wait()和notify()

线程的状态

状态是针对当前线程调度的情况来描述的,也可以认为线程是调度的基本单位,在Java中对线程的状态进行了细化,主要分为下面几种状态

★ 1.NEW创建了Thread对象,但是还没有调用start

★ 2. TERMINATED 表示内核中的pcb已经执行完了,但是Thread对象还在.

★ 3. RUNNABLE 可运行状态,分为两种情况,一种是正在cpu上执行的,另外一种是在就绪队列里面,随时可以去CPU上执行的

★ 4.WAITING,TIMED_WAITING,BLOCKED 都是阻塞状态,都表示线程PCB正在阻塞队列里面.但触发条件不同,.WAITING是wait()方法触发的,TIMED_WAITING是sleep()方法触发的,BLOCKED是synchronized触发的.synchronized的相关操作在另外一篇博客里

synchronized的相关操作

wait()和notify()

线程最大的问题是抢占式执行,随机调度,但为了保证线程的安全,不希望随机调度,程序员就想出一些办法,控制现成的执行顺序,虽然线程在内核中的调度是随机的,但可以通过一些api让线程主动阻塞,主动放弃cpu.

比如有t1,t2两个线程,我们希望t1先执行一半,然后让t2执行,就可以让t2先wait(进入阻塞状态,主动放弃cpu),等t1执行执行一半的时候,在通过notify来通知t2,把t2唤醒,让t2开始执行

wait进行阻塞,某个线程调用wait方法(无论是那个对象),此时就会进入阻塞,此时就处于WAITING状态.

用代码来看看wait的使用方法

运行结果

此时抛出非法的锁状态异常,锁的状态就是被加锁的状态和被解锁的状态.

为什么胡出现这种异常呢,来看看wait的具体操作

1. 先释放锁

2. 进行阻塞等待

3. 收到通知后,重新尝试获取锁,并在获取锁后,继续往下执行

对上面代码进行修改,wait要先释放锁,就要先加锁,我们对上面代码进行加锁操作

运行结果

此时代码阻塞在synchronized代码块里,实际上这里的阻塞是释放了锁,然后进行阻塞等待,此时就处于WAITING 状态,

此时就需要notify来进行通知,然后唤醒wait,同样,notify()和wait()一样,在使用之前要先释放锁.

在对上面代码进行改进

代码的执行结果

在wait()的无参数版本,只要t2不进行notify(),t1就会一直等待下去

在java中wait()还有带参数的版本,可以指定等待的最大时间

还有一个唤醒wait的方法,notifyAll(),此时,当多个线程wait的时候,notify是随机唤醒一个线程,而notifyAll()是唤醒所有线程,这些线程在一起竞争锁.

相关推荐
kyle~3 分钟前
python---PyInstaller(将Python脚本打包为可执行文件)
开发语言·前端·python·qt
望获linux5 分钟前
【实时Linux实战系列】规避缺页中断:mlock/hugetlb 与页面预热
java·linux·服务器·数据库·chrome·算法
User:你的影子9 分钟前
WPF ItemsControl 绑定
开发语言·前端·javascript
失散1327 分钟前
分布式专题——9 Redis7底层数据结构解析
java·数据结构·redis·分布式·缓存·架构
馨谙32 分钟前
设计模式之单例模式大全---java实现
java·单例模式·设计模式
程序员TNT35 分钟前
Shoptnt 安全架构揭秘:JWT 认证与分布式实时踢人方案
java·redis·分布式·架构
野生的编程萌新37 分钟前
【C++深学日志】从0开始的C++生活
c语言·开发语言·c++·算法
水无痕simon40 分钟前
3 水平分表
java·数据库
野生程序员y1 小时前
深入解析Spring AOP核心原理
java·后端·spring
阿萨德528号1 小时前
ZooKeeper Java客户端与分布式应用实战
java·zookeeper·java-zookeeper