线程通信和两个经典例题(两个线程交替打印1-100的数字+生产者消费者问题)

线程通信

概述

1、线程通讯指的是多个线程通过消息传递实现相互牵制,相互调度,即线程间的相互作用。

2、线程通信中涉及的三个方法:

(1)wait():此方法一旦执行,当前线程就进入阻塞状态,并释放同步监视器(锁资源)。

(2)notify():此方法一旦执行,就会唤醒被 wait 的一个线程。如果有多个线程被 wait,就唤醒优先级最高的那个。

(3)notifyAll():此方法一旦执行,就会唤醒所有被 wait 的线程。

3、wait()、notify()、notifyAll()三个方法必须使用在同步代码块或者同步方法中。

4、wait()、notify()、notifyAll()三个方法是定义在java.lang.Object类中。

线程通信的经典案例
两个线程交替打印1-100的数字
public class ThreadComunicationDemo {
    /*这是一个程序入口*/
    public static void main(String[] args) {
        Comunication c = new Comunication();
        Thread t1 = new Thread(c,"线程1");
        Thread t2 = new Thread(c,"线程2");
        t1.start();
        t2.start();
    }
}


class Comunication implements Runnable{

    /*共享资源*/
    int num = 1;

    @Override
    public void run() {
        while(true){
            synchronized (this) {
                /*当以个进入wait后就释放了锁,此时第二个线程拿到锁后并释放被wait的线程*/
                this.notify();
                if (num <= 100) {
                    System.out.println(Thread.currentThread().getName() + ":" + num);
                    num++;
                } else {
                    break;
                }
                /*一个线程输出一个数字后就自行进入wait*/
                try {
                    this.wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}
生产者消费者问题

问题描述:

生产者(Productor)将产品放在柜台(Counter),而消费者(Customer)从柜台处取走产品,生产者一次只能生产固定数量的产品(比如:1), 这时柜台中不能再放产品,此时生产者应停止生产等待消费者拿走产品,此时生产者唤醒消费者来取走产品,消费者拿走产品后,唤醒生产者,消费者开始等待.

/**
 * @function 生产者的线程
 */
public class Productor implements Runnable {
    /*共享资源*/
    private Counter counter;//不能直接对柜台进行实例化,因为消费者和生产者同时实例柜台,他们的商品库存就是分别为1,所以需要在main方法中实例化,然后传到两个线程

    public Productor(Counter counter) {
        this.counter = counter;
    }

    @Override
    public void run() {
        //生产者生产物品
        while(true){
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            counter.getNum();
        }
    }
}

/**
 * @function 消费者线程
 */
public class Customer implements Runnable {
    //共享资源
    private Counter counter;

    public Customer(Counter counter) {
        this.counter = counter;
    }

    @Override
    public void run() {
        //消费者消费物品
        while (true) {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            counter.buyNum();
        }
    }
}

/**
 * @function 柜台 两个线程的共享资源 库存永远保持1
 */
public class Counter {
    private static int num = 1; //库存数量

    /**
     * 消费者买一个商品,库存-1
     */
    public synchronized void buyNum(){
        //库存为0时,让消费者等待
        if (num==0){
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }else{
            //库存不为0时,此时从商品柜台买走一个物品,唤醒生产者的线程,生产者生产东西
            System.out.println(Thread.currentThread().getName()+"买了1件物品");
            num-=1;
            notify();
        }
    }

    /**
     * 生产者生产一个商品,库存+1
     */
    public synchronized void getNum(){
        //库存为1时,让生产者等一等,不要生产东西
        if (num==1){
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }else{
            //库存不满时,生产者生产,同时唤醒消费者线程买东西
            System.out.println(Thread.currentThread().getName()+"生产1件物品");
            num+=1;
            notify();
        }

    }
}


/**
 * @function
 */
public class Test {
    /*这是一个程序入口*/
    public static void main(String[] args) {
        /*共享资源*/
        Counter counter = new Counter();

        Productor productor = new Productor(counter);
        Thread thread = new Thread(productor,"生产者线程");
        thread.start();
        Customer customer = new Customer(counter);
        Thread thread1 = new Thread(customer,"消费者线程");
        thread1.start();
    }
}
  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值