首页 > 其他分享 >为什么 wait/notify 必须与 synchronized 一起使用

为什么 wait/notify 必须与 synchronized 一起使用

时间:2023-03-20 18:26:18浏览次数:42  
标签:obj synchronized lock 线程 notify wait

注:本文转自:https://mp.weixin.qq.com/s/ZbB_4vYg6aNMDyy6KnjSnA

为什么 java wait/notify 必须与 synchronized 一起使用

这个问题就是书本上没怎么讲解,就是告诉我们这样处理,但没有解释为什么这么处理?我也是基于这样的困惑去了解原因。

synchronized是什么

Java中提供了两种实现同步的基础语义:synchronized方法和synchronized块, 看个demo:

public class SyncTest {

   \\ 1、synchronized方法
   
 public synchronized void syncMethod(){
        System.out.println("hello method");
    }
    
    \\ 2、synchronized块
    
    public void syncBlock(){
        synchronized (this){
            System.out.println("hello block");
        }
    }
}

具体还要区分:

  • 修饰实例方法,作用于当前实例加锁,进入同步代码前要获得当前实例的锁。不同实例对象的访问,是不会形成锁的。
  • 修饰静态方法,作用于当前类对象加锁,进入同步代码前要获得当前类对象的锁
  • 修饰代码块,指定加锁对象,对给定对象加锁,进入同步代码库前要获得给定对象的锁。

它具有的特性:

  • 原子性
  • 可见性
  • 有序性
  • 可重入性

synchronized如何实现锁

这样看来synchronized实现的锁是基于class对象来实现的,我们来看看如何实现的,它其实是跟class对象的对象头一起起作用的,对象在内存中的布局分为三块区域:对象头、实例数据和对齐填充。

其中对象头中有一个Mark Word,这里主要存储对象的hashCode、锁信息或分代年龄或GC标志等信息,把可能的情况列出来大概如下:

 

 

其中synchronized就与锁标志位一起作用实现锁。主要分析一下重量级锁也就是通常说synchronized的对象锁,锁标识位为10,其中指针指向的是monitor对象(也称为管程或监视器锁)的起始地址。

每个对象都存在着一个 monitor 与之关联,对象与其 monitor 之间的关系有存在多种实现方式,如monitor可以与对象一起创建销毁或当线程试图获取对象锁时自动生成,但当一个 monitor 被某个线程持有后,它便处于锁定状态。

在Java虚拟机(HotSpot)中,monitor是由ObjectMonitor实现的,其主要数据结构如下(位于HotSpot虚拟机源码ObjectMonitor.hpp文件,C++实现的):

ObjectMonitor() {
    _header       = NULL;
    _count        = 0; //记录个数
    _waiters      = 0,
    _recursions   = 0;
    _object       = NULL;
    _owner        = NULL;
    _WaitSet      = NULL; //处于wait状态的线程,会被加入到_WaitSet
    _WaitSetLock  = 0 ;
    _Responsible  = NULL ;
    _succ         = NULL ;
    _cxq          = NULL ;
    FreeNext      = NULL ;
    _EntryList    = NULL ; //处于等待锁block状态的线程,会被加入到该列表
    _SpinFreq     = 0 ;
    _SpinClock    = 0 ;
    OwnerIsThread = 0 ;
  }

上面有2个字段很重要:

  • _WaitSet队列处于wait状态的线程,会被加入到_WaitSet。
  • _EntryList队列处于等待锁block状态的线程,会被加入到该列表。
  • _owner_owner指向持有ObjectMonitor对象的线程

我们来模拟一下进入锁的流程:

1、当多个线程同时访问一段同步代码时,首先会进入 _EntryList 集合

2、当线程获取到对象的monitor 后进入 _Owner 区域,并把monitor中的owner变量设置为当前线程同时monitor中的计数器count加1

3、若线程调用 wait() 方法,将释放当前持有的monitor,owner变量恢复为null,count自减1,同时该线程进入 WaitSet集合中等待被唤醒。

4、若当前线程执行完毕也将释放monitor(锁)并复位变量的值,以便其他线程进入获取monitor(锁)

wait/notify

这两个是Java对象都有的属性,表示这个对象的一个等待和通知机制,

不用synchronized 会怎么样

参考其他博客,我们来看看不使用synchronized会怎么样,假设有2个线程,分别做2件事情,T1线程代码逻辑:

while(!条件满足) // line 1  
{  
    obj.wait(); // line 2  
}  
doSomething();  

T2线程的代码逻辑:

更改条件为满足; // line 1  
obj.notify(); // line 2 

多线程环境下没有synchronized,没有锁的情况下可能会出现如下执行顺序情况:

  • T1 line1 满足while 条件
  • T2 line1 执行
  • T2 line2 执行,notify发出去了
  • T1 line2 执行,wait再执行

这样的执行顺序导致了notify通知发出去了,但没有用,已经wait是在之后执行,所以有人说没有保证原子性,就是line1 和line2 是一起执行结束,这个也被称作lost wake up问题。解决方法就是可以利用synchronized来加锁,于是有人就写了这样的代码:

synchronized(lock)  
{  
    while(!条件满足)  
    {  
        obj.wait();  
    }  
    doSomething();  
}  
synchronized(lock)  
{  
   更改条件为满足;  
   obj.notify();  
}  

这样靠锁来做达到目的。但这代码会造成死锁,因为先T1 wait(),再T2 notify();而问题在于T1持有lock后block住了,T2一直无法获得lock,从而永无可能notify()并将T1的block状态解除,就与T1形成了死锁。

所以JVM在实现wait()方法时,一定需要先隐式的释放lock,再block,并且被notify()后从wait()方法返回前,隐式的重新获得了lock后才能继续user code的执行。要做到这点,就需要提供lock引用给obj.wait()方法,否则obj.wait()不知道该隐形释放哪个lock,于是调整之后的结果如下:

synchronized(lock)  
{  
    while(!条件满足)  
    {  
        obj.wait(lock);  
        // obj.wait(lock)伪实现  
        //   [1] unlock(lock)  
        //   [2] block住自己,等待notify()  
        //   [3] 已被notify(),重新lock(lock)  
        //   [4] obj.wait(lock)方法成功返回  
    }  
    doSomething();  
}  

[最终形态] 把lock和obj合一

其它线程API如PThread提供wait()函数的签名是类似cond_wait(obj, lock)的,因为同一个lock可以管多个obj条件队列。而Java内置的锁与条件队列的关系是1:1,所以就直接把obj当成lock来用了。因此此处就不需要额外提供lock,而直接使用obj即可,代码也更简洁:

synchronized(obj)  
{  
    while(!条件满足)  
    {  
        obj.wait();  
    }  
    doSomething();  
} 
synchronized(lock)  
{  
   更改条件为满足;  
   obj.notify();  
}  

lost wake up

wait/notify 如果不跟synchronized结合就会造成lost wake up,难以唤醒wait的线程,所以单独使用会有问题。

标签:obj,synchronized,lock,线程,notify,wait
From: https://www.cnblogs.com/wk-missQ1/p/17237240.html

相关文章

  • Android RecyclerView的notify方法和动画的刷新详解
    前些天发现了一个蛮有意思的人工智能学习网站,8个字形容一下"通俗易懂,风趣幽默",感觉非常有意思,忍不住分享一下给大家。......
  • golang  实现 sync.WaitGroup wait() 方法 超时 自动释放
    思路是把wg.wait()放到一个协程里,通过chan向外发送完成信号。外层通过一个select超时结构来控制最大超时时间。funcwaitTimeout(wg*sync.WaitGroup,timeouttime.Du......
  • synchronized 和 Lock 的区别
    Lock是一个接口,而synchronized是Java中的关键字,synchronized是内置的语言实现;synchronized在发生异常时,会自动释放线程占有的锁,因此不会导致死锁现象发生;而Lock......
  • 使用synchronized对并发性的影响
    1前言非静态方法的同步锁是当前对象(this)(对象锁)静态方法的同步锁是当前类的字节码(类锁)不同的锁之间能并发2同一对象内本节主类与资源类如下:classResorce{//资源......
  • Java synchronized的实现原理
    通常在多线程执行的过程中,我们需要考虑一些线程安全的问题,而线程安全问题中最常用的解决策略之一就是“锁”。加锁的本质,就是为了解决在多线程场景中对于共享数据访问的......
  • mysql小知识点---interactive_timeout和wait_timeout区别
    interactive_timeout定义了对于交互式连接(比如使用cmd命令窗口或者在linux上连接msyql),服务器等待的最大时间,如果超过这个时间,服务端仍然没有受到数据,则会关闭连接;【我理解......
  • 启动vagrant up 报错 `await_response_state': scp: /tmp/vagrant-network-entry-eth1
      解决办法Linux df命令用于显示目前在Linux系统上的文件系统的磁盘使用情况统计。Linuxdu命令用于显示目录或文件的大小。du会显示指定的目录或文件所占用的磁盘......
  • 谈谈线程间的协作(wait/notify/sleep/yield/join)
    线程的状态Java中线程中状态可分为五种:New(新建状态),Runnable(就绪状态),Running(运行状态),Blocked(阻塞状态),Dead(死亡状态)。New:新建状态,当线程创建完成时为新建状态,即newThread......
  • 对并发熟悉吗?说说synchronized及实现原理
    synchronized的基本使用synchronized是Java中解决并发问题的一种最常用的方法,也是最简单的一种方法。作用主要有三个:确保线程互斥的访问同步代码保证共享变量的修改能......
  • Rocky Linux 系列6 --- inotify-tools
    一、概念1.环境(1)RockyLinux9.1(2)inotify-tools3.22.1.02.概念inotify用于监控文件或目录的改变,当文件或目录发生改变时inotify将触发通知。3.承上启下(1......