博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
为什么wait()和notify()属于Object类
阅读量:4683 次
发布时间:2019-06-09

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

关于wait()暂停的是持有锁的对象,所以想调用wait()必须为:对象.wait();

notify()唤醒的是等待锁的对象,调用:对象.notify();

如下:

Object obj = newObject();

synchronized(obj){

    try{  

      obj.wait();

      }catch(Exception e){}

      obj.notify();

  }

注意:wait(),notify(),notifyAll()都必须使用在同步中,因为要对持有监视器(锁)的线程操作。所以要使用在同步中,因为只有同步 才具有锁。

为什么这些操作线程的方法要定义在object类中呢?

简单说:因为synchronized中的这把锁可以是任意对象,所以任意对象都可以调用wait()和notify();所以wait和notify属于Object。

专业说:因为这些方法在操作同步线程时,都必须要标识它们操作线程的锁,只有同一个锁上的被等待线程,可以被同一个锁上的notify唤醒,不可以对不同锁中的线程进行唤醒。

也就是说,等待和唤醒必须是同一个锁。而锁可以是任意对象,所以可以被任意对象调用的方法是定义在object类中。

 

在jdk1.5以后,将同步synchronized替换成了Lock,将同步锁对象换成了Condition对象,并且Condition对象可以有多个,这样可以解决一个问题。

比如说我们在多个生产者和消费者模式中:

boolean flag = false;

public synchronized void set(String name){

  while(flag){//用while而不用if的原因,这样每个线程在wait等待醒来后都必须再次判断flag

    try{this.wait();}catch(Exception e){}  

    Sytem.out.printLn("生产者");

    flag = true;

    this.notifyAll();//这将唤醒所有线程(本方线程和对方线程),消耗资源

  }

}

public synchronized void out(){

  whie(!flag){

     try{this.wait();}catch(Exception e){}

    Sytem.out.printLn("消费者");

    flag = false;

    this.notifyAll();//这将唤醒所有线程(本方线程和对方线程),消耗资源

  }

}

上面的做法很消耗资源,如果把notifyAll()改成notify()的话,就会造成可能所有线程都在等待

 

所以在jdk1.5以后提供了Lock接口和Condition对象。Condition中的await(), signal().signalAll()代替Object中的wait(),notify(),notifyAll()

private Lock lock = new ReentrantLock();

private Condition condition_pro = lock.newCondition();//生产者对象

private Condition condition_con = lock.newCondition();//消费者对象

public void set(String name) throws Exception{

  lock.lock();//加锁

  try{

    while(flag){

     contion_pro.await();

     Sytem.out.printLn("生产者");

       flag= true;

     condition_con.singal();//指定唤醒消费方

   }finally{

    lock.unlock();//解锁    

    }  

  }

}

public void out() throws Exception{

  lock.lock();

  try{

    while(!flag){

       condition_con.await(); 

       Sytem.out.printLn("消费者");

       flag = false;

       condition_pro.signal();//指定唤醒生产方

      }finally{

      lock.unlock();  

      }

    }

}

这样做的好处,我们可以指定唤醒某一方,减少消耗

 

Java小生店铺:

Pc端:

手机端:搜索 java小生店铺

希望店铺的资料能帮助到你!!!

 

转载于:https://www.cnblogs.com/lirenzhujiu/p/5927241.html

你可能感兴趣的文章
MySQL批量SQL插入性能优化
查看>>
定义列属性:null,default,PK,auto_increment
查看>>
用户画像展示
查看>>
pyqt pyinstaller使用说明
查看>>
C#中StreamReader读取中文出现乱码
查看>>
引用堆中的对象
查看>>
用CSS开启硬件加速来提高网站性能(转)
查看>>
使用BufferedReader的时候出现的问题
查看>>
加快页面加载速度的方法
查看>>
Oozie协作框架
查看>>
linux安装图形界面
查看>>
Android广播发送失败
查看>>
博弈论之入门小结
查看>>
解决IE8下opacity属性失效问题,无法隐藏元素
查看>>
洛谷1002 过河卒
查看>>
C#匿名函数的坑
查看>>
标记页面控件尺寸
查看>>
批处理文件中的路径问题
查看>>
appium+python 环境搭建
查看>>
WampServer下修改和重置MySQL密码
查看>>