【Java基础】-- 线程同步举例:生产者消费者(通过Object类提供的wait()、notify()、notifyAll()实现)

一 线程同步和互斥

1 互斥:是指某一个资源某个时刻只允许一个访问者对其进行访问,具有唯一性和排他性。

2 同步:在互斥的基础上,通过其他机制实现访问者对资源的有序访问,通过一定的逻辑关系来共同完成一个任务。

3 同步和互斥的联系:同步其实已经实现了互斥,所以同步是一种更为复杂的互斥。

4 同步和互斥的区别:

(1)互斥是不同线程通过竞争进入临界区(共享的数据和硬件资源),为了防止冲突,在同一时刻只允许一个线程使用共享资源,如不能同时写。

(2)同步时多个线程彼此合作,通过一定的逻辑关系来共同完成一个任务,如先生产再使用。

(3)简单来说,互斥时通过竞争实现对资源的独占使用,彼此之间不需要知道对方的存在,执行顺序是一个乱序,同步时协调多个相互关联的线程完成任务,彼此之间知道对方存在,执行顺序往往是有序的。

 

二 实现同步的方法

1 实现同步的常见方法

(1)通过Object的wait()、notify()和notifyAll()。

(2)通过Condition的awiat()、signal()和singalAll()。

(3)通过阻塞队列(BlockingQueue)。

2 生产者消费者线程同步举例(通过Object的wait()和notify()实现)

(1)产品类

public class Product {
    private int number = 0;
    public synchronized void increase() {
        if (number < 20) {
            number++;
            System.out.println(Thread.currentThread().getName()
                + ": 开始生产第" + number + "个产品。");
            notify();
        } else {
            try {
                wait();
                System.out.println("生产满了,休息一下。");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public synchronized void decrease() {
        if (number > 0) {
            System.out.println(Thread.currentThread().getName()
                + ": 开始消费第" + number + "个产品。");
            number--;
            notify();
        } else {
            try {
                wait();
                System.out.println("没有商品了");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

(2)生产者类

public class Producer extends Thread {

    private Product product;

    public Producer(Product product) {
        this.product = product;
    }

    @Override
    public void run() {
        System.out.println(getName() + ":开始生产产品。。。。");
        for (int i = 0; i < 50; i++) {
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            product.increase();
        }
    }
}

(3)消费者类

public class Consumer extends Thread {

    private Product product;

    public Consumer(Product product) {
        this.product = product;
    }

    @Override
    public void run() {
        System.out.println(getName() + ":开始消费产品。。。。");
        for (int i = 0; i < 50; i++) {
            try {
                Thread.sleep(150);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            product.decrease();
        }
    }
}

(4)测试类

public class Test {
    public static void main(String[] args) {
        Product product = new Product();
        Producer p1 = new Producer(product);
        p1.setName("生产者1");
        Producer p2 = new Producer(product);
        p2.setName("生产者2");

        Consumer c1 = new Consumer(product);
        c1.setName("消费者1");
        Consumer c2 = new Consumer(product);
        c2.setName("消费者2");

        p1.start();
        p2.start();
        c1.start();
        c2.start();
    }
}

(5)运行结果

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
生产者消费者问题是一种经典的多线程同步问题,主要涉及到生产者和消费者之间的数据交换和同步。 在Java,可以使用多种方式来实现生产者消费者问题的解决方案,其最常用的方式是使用wait()和notify()方法来实现线程间的同步。 下面是一个简单的示例,演示了如何使用wait()和notify()方法来实现生产者消费者问题的解决方案: ``` import java.util.LinkedList; import java.util.Queue; public class ProducerConsumer { private Queue<Integer> buffer = new LinkedList<>(); private final int capacity = 5; public void produce() throws InterruptedException { int value = 0; while (true) { synchronized (this) { while (buffer.size() == capacity) wait(); System.out.println("Producer produced-" + value); buffer.add(value++); notify(); Thread.sleep(1000); } } } public void consume() throws InterruptedException { while (true) { synchronized (this) { while (buffer.size() == 0) wait(); int val = buffer.poll(); System.out.println("Consumer consumed-" + val); notify(); Thread.sleep(1000); } } } } ``` 在上面的示例,ProducerConsumer表示生产者消费者问题的解决方案。这个有一个buffer队列,它是一个FIFO队列,用于存储生产者生产的数据。 produce()方法表示生产者的行为,它使用一个while(true)循环来不断地生产数据。在每次生产之前,它首先检查buffer队列是否已满。如果buffer队列已满,则调用wait()方法来等待消费者线程消费数据。如果buffer队列未满,则将生产的数据添加到buffer队列,并调用notify()方法通知消费者线程可以消费数据了。 consume()方法表示消费者的行为,它使用一个while(true)循环来不断地消费数据。在每次消费之前,它首先检查buffer队列是否为空。如果buffer队列为空,则调用wait()方法来等待生产者线程生产数据。如果buffer队列不为空,则从buffer队列取出一个数据,并调用notify()方法通知生产者线程可以生产数据了。 在上面的示例,使用了synchronized关键字来实现线程的同步。synchronized关键字可以确保在同一时间只有一个线程可以执行被synchronized关键字包裹的代码块。wait()方法可以让线程等待,直到被notify()方法唤醒。notify()方法可以唤醒一个等待的线程。如果有多个线程等待,则只有一个线程会被唤醒。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值