锁 唤醒_生产者消费者之等待唤醒机制

API ---- 多线程 ---- 线程间通信

生产者消费者之等待唤醒机制:

f31a597fa0f1fe4d1a64d6cdbe7aaf7b.png
等待唤醒机制思路图解.bmp
* 问题3:虽然数据安全了,但是呢,一次一大片不好看,我就想依次的一次一个输出。
 * 如何实现呢?
 *      通过Java提供的等待唤醒机制解决。
 * 
 * 等待唤醒:
 *      Object类中提供了三个方法:
 *          wait():等待,并立即释放锁,将来还会在这里唤醒,而不是从头走代码
 *          notify():唤醒单个线程
 *          notifyAll():唤醒所有线程
 *      为什么这些方法不定义在Thread类中呢?
 *          这些方法的调用必须通过锁对象调用,而我们刚才使用的锁对象是任意锁对象。
 *          所以,这些方法必须定义在Object类中。

生产者消费者之等待唤醒机制:

演示3:等待唤醒机制1

package cn.itcast_05;
​
/*
 * 分析:
 *      资源类:Student 
 *      设置学生数据:SetThread(生产者)
 *      获取学生数据:GetThread(消费者)
 *      测试类:StudentDemo
 * 
 * 问题1:按照思路写代码,发现数据每次都是:null---0
 * 原因:我们在每个线程中都创建了新的资源,而我们要求的时候设置和获取线程的资源应该是同一个
 * 如何实现呢?
 *      在外界把这个数据创建出来,通过构造方法传递给其他的类。
 * 
 * 问题2:为了数据的效果好一些,我加入了循环和判断,给出不同的值,这个时候产生了新的问题
 *      A:同一个数据出现多次
 *      B:姓名和年龄不匹配
 * 原因:
 *      A:同一个数据出现多次
 *          CPU的一点点时间片的执行权,就足够你执行很多次。
 *      B:姓名和年龄不匹配
 *          线程运行的随机性
 * 线程安全问题:
 *      A:是否是多线程环境      是
 *      B:是否有共享数据       是
 *      C:是否有多条语句操作共享数据 是
 * 解决方案:
 *      加锁。
 *      注意:
 *          A:不同种类的线程都要加锁。
 *          B:不同种类的线程加的锁必须是同一把。
 * 
 * 问题3:虽然数据安全了,但是呢,一次一大片不好看,我就想依次的一次一个输出。
 * 如何实现呢?
 *      通过Java提供的等待唤醒机制解决。
 * 
 * 等待唤醒:
 *      Object类中提供了三个方法:
 *          wait():等待
 *          notify():唤醒单个线程
 *          notifyAll():唤醒所有线程
 *      为什么这些方法不定义在Thread类中呢?
 *          这些方法的调用必须通过锁对象调用,而我们刚才使用的锁对象是任意锁对象。
 *          所以,这些方法必须定义在Object类中。
 */
public class StudentDemo {
    public static void main(String[] args) {
        //创建资源
        Student s = new Student();
        
        //设置和获取的类
        SetThread st = new SetThread(s);
        GetThread gt = new GetThread(s);
​
        //线程类
        Thread t1 = new Thread(st);
        Thread t2 = new Thread(gt);
​
        //启动线程
        t1.start();
        t2.start();
    }
}
package cn.itcast_05;
​
public class SetThread implements Runnable {
​
    private Student s;
    private int x = 0;
​
    public SetThread(Student s) {
        this.s = s;
    }
​
    @Override
    public void run() {
        while (true) {
            synchronized (s) {
                //判断有没有
                if(s.flag){
                    try {
                        s.wait(); //t1等着,释放锁
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                
                if (x % 2 == 0) {
                    s.name = "林青霞";
                    s.age = 27;
                } else {
                    s.name = "刘意";
                    s.age = 30;
                }
                x++; //x=1
                
                //修改标记
                s.flag = true;
                //唤醒线程
                s.notify(); //唤醒t2,唤醒并不表示你立马可以执行,必须还得抢CPU的执行权。
            }
            //t1有,或者t2有
        }
    }
}
package cn.itcast_05;
​
public class GetThread implements Runnable {
    private Student s;
​
    public GetThread(Student s) {
        this.s = s;
    }
​
    @Override
    public void run() {
        while (true) {
            synchronized (s) {
                if(!s.flag){
                    try {
                        s.wait(); //t2就等待了。立即释放锁。将来醒过来的时候,是从这里醒过来的时候
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                
                System.out.println(s.name + "---" + s.age);
                //林青霞---27
                //刘意---30
                
                //修改标记
                s.flag = false;
                //唤醒线程
                s.notify(); //唤醒t1
            }
        }
    }
}
package cn.itcast_05;
​
public class Student {
    String name;
    int age;
    boolean flag; // 默认情况是没有数据,如果是true,说明有数据
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值