线程的生命周期 |
线程同步 |
为什么要有线程同步?
因为普通线程争抢资源会出现线程安全问题。以3个售票机同时卖100张票为例,此时,3个售票机是3个线程,100张票是共享资源。我们的逻辑是:如果余票大于0,则进入系统,否则,结束。
多卖出现的原因:
(1)当只剩下最后1张时,线程1判断还有1张,进入系统,但是没等输出,cpu资源就被抢夺了。
(2) 此时2进行判断,发现还有1张,进入系统,依旧没等输出,cpu资源就被抢夺了。
(3) 3同样。此时3个线程都处于阻塞状态。
(4)然后,1醒了,开始执行输出,此时余票为0。
(5)2醒了,输出,此时余票为-1。
(6)同理,3输出,余票为-2。
从上面的示例中我们得出:
- 线程安全问题存在的原因是:由于一个线程在操作共享数据过程中,未执行完毕的情况下,另外的线程参与进来,导致共享数据存在了安全问题。
- 解决办法是:必须让一个线程操作共享数据完毕以后,其他线程才有机会参与共享数据的操作。也就是用线程的同步机制。
什么是线程同步?
线程排队,几个线程之间要排队,一个一个对共享资源进行操作,而不是同时进行操作。就像上厕所,总得一个一个来,如果几个人都进去,那肯定会出问题。
什么时候用线程同步?
1、只有对共享资源的读写访问才需要同步,就像如果你上男厕,她上女厕,你俩不会出现争抢问题。
2、只有“变量”才需要同步,票的数量是变动的,所以它需要同步。如果几个线程同时读取同一个文件,那自己读自己的,互不影响,这个就不需要运用同步机制。
综上:如果几个线程需要同时访问同一份可变的共享资源,那就需要线程同步。
如何实现线程同步?
(1)同步代码块

调用:
运行结果,从结果中发现,已没有重票和错票的问题了。
(2)同步方法

线程的通信 |
用到的方法在线程声明周期的图中也能看到。
方法 | 解释 |
---|---|
wait() | 令当前线程挂起并放弃cpu、同步资源,使别的线程可访问并修改共享资源,而当前线程排队等候再次对资源的访问。 |
notify() | 唤醒正在排队等待同步资源的线程中优先级最高者结束等待 |
notifyAll() | 唤醒正在排队等待资源的所有线程结束等待。 |
备注:只能在synchronized方法或synchronized代码块中才能使用,否则会报java.lang.lllegalMonitorStateException异常。
参考博客:https://blog.youkuaiyun.com/u012179540/article/details/40685207