博客
关于我
Java 并发之虚假唤醒
阅读量:327 次
发布时间:2019-03-04

本文共 2594 字,大约阅读时间需要 8 分钟。

目录

 


一、什么是虚假唤醒?

当一个条件满足时,很多线程都被唤醒了,但是只有其中部分是有用的唤醒,其它的唤醒都是无用功1.比如说买货,如果商品本来没有货物,突然进了一件商品,这是所有的线程都被唤醒了	,但是只能一个人买,所以其他人都是假唤醒,获取不到对象的锁

二、开发中虚假唤醒会造成问题的场景:

例如在多线程的场景中,判断一个线程是否需要阻塞(调用wait())时,如果用 if 判断,就会出问题,因为想想看,多有多个线程都在阻塞时,突然条件满足了(product<1),那么就会有线程唤醒阻塞着的多个线程,但是只有一个线程能拿到锁,其余没拿到锁的线程要是被唤醒了,还继续往下执行,就不对了。

if(product>=1){                wait();        }

所以解决方法是 调用wait的代码块,要用while 来代替 if :

while(product>=1){                wait();        }

三、为什么 if 会产生虚假唤醒

因为if只会执行一次,执行完会接着向下执行if()外边的 而while不会,直到条件满足才会向下执行while()外边的

四、怎么产生虚假唤醒

把 while (product >=1) {} 换成 if (product >=1) {} 就会出现虚假唤醒

 

下面给个 生产者消费者的代码展示:

public class Test {    public static void main(String[] args) {        Clerk clerk = new Clerk();        Producer producer = new Producer(clerk);        Customer customer = new Customer(clerk);        new Thread(producer,"生产者A").start();        new Thread(customer,"消费者A").start();        new Thread(producer,"生产者B").start();        new Thread(customer,"消费者B").start();    }}class Clerk{    private int product=0;    public synchronized void add(){        while(product>=1){            System.out.println(Thread.currentThread().getName()+"---还有货物:"+product);            try {                wait();            } catch (InterruptedException e) {                e.printStackTrace();            }        }        product+=1;        System.out.println(Thread.currentThread().getName()+" 补货完成,剩余货物:"+product);        notifyAll();    }    public synchronized void sale(){        while (product<=0){            System.out.println(Thread.currentThread().getName()+" 缺货中:"+product);            try {                wait();            } catch (InterruptedException e) {                e.printStackTrace();            }        }        System.out.println(Thread.currentThread().getName()+" 已补货,开始消费...");        product-=1;        notifyAll();    }}class Producer implements Runnable{    private Clerk clerk;    public Producer(Clerk clerk){        this.clerk = clerk;    }    @Override    public void run() {        for (int i=0;i<20;i++){            try {                Thread.sleep(200);            } catch (InterruptedException e) {                e.printStackTrace();            }            clerk.add();        }    }}class Customer implements Runnable{    private Clerk clerk;    public Customer(Clerk clerk){        this.clerk = clerk;    }    @Override    public void run() {        for (int i = 0; i < 20; i++) {            try {                Thread.sleep(200);            } catch (InterruptedException e) {                e.printStackTrace();            }            clerk.sale();        }    }}

 

转载地址:http://lofh.baihongyu.com/

你可能感兴趣的文章
Mysql中varchar类型数字排序不对踩坑记录
查看>>
MySQL中一条SQL语句到底是如何执行的呢?
查看>>
MySQL中你必须知道的10件事,1.5万字!
查看>>
MySQL中使用IN()查询到底走不走索引?
查看>>
Mysql中使用存储过程插入decimal和时间数据递增的模拟数据
查看>>
MySql中关于geometry类型的数据_空的时候如何插入处理_需用null_空字符串插入会报错_Cannot get geometry object from dat---MySql工作笔记003
查看>>
mysql中出现Incorrect DECIMAL value: '0' for column '' at row -1错误解决方案
查看>>
mysql中出现Unit mysql.service could not be found 的解决方法
查看>>
mysql中出现update-alternatives: 错误: 候选项路径 /etc/mysql/mysql.cnf 不存在 dpkg: 处理软件包 mysql-server-8.0的解决方法(全)
查看>>
Mysql中各类锁的机制图文详细解析(全)
查看>>
MySQL中地理位置数据扩展geometry的使用心得
查看>>
Mysql中存储引擎简介、修改、查询、选择
查看>>
Mysql中存储过程、存储函数、自定义函数、变量、流程控制语句、光标/游标、定义条件和处理程序的使用示例
查看>>
mysql中实现rownum,对结果进行排序
查看>>
mysql中对于数据库的基本操作
查看>>
Mysql中常用函数的使用示例
查看>>
MySql中怎样使用case-when实现判断查询结果返回
查看>>
Mysql中怎样使用update更新某列的数据减去指定值
查看>>
Mysql中怎样设置指定ip远程访问连接
查看>>
mysql中数据表的基本操作很难嘛,由这个实验来带你从头走一遍
查看>>