线程安全的几种方法

本文探讨了在Java中处理多线程并发时如何确保线程安全。介绍了同步代码块、同步方法和Lock锁机制这三种方法,并对比了synchronized与ReentrantLock的优缺点。强调在并发量小时synchronized适用,而高并发时推荐使用ReentrantLock。同时,阐述了wait()、notify()和notifyAll()在线程通信中的作用和使用注意事项。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

通常情况下,一个进程中比较耗时的操作有文件上传下载,长循环,网络资源的获取,这个时候往往会采用多线程的方式来处理。

当进程中有多个并发线程在执行,并且需要处理一个数据代码块是,往往需要考虑线程安全问题,如果线程不安全的话,可能造成数据不一致的问题。

第一种实现线程安全的方式:同步代码块,即用synchronized关键字

第二种方法:同步方法,也是用synchronized关键字,只是这个关键字用在方法上了,把线程共享的数据块抽象成方法,在方法上加了同步锁。

第三种方法:使用Lock锁机制,对线程不安全的代码块采用lock()加锁,使用unlock()解锁。

总结:

        由于synchronized是在JVM层面实现的,因此系统可以监控锁的释放与否;而ReentrantLock是使用代码实现的,系统无法自动释放锁,需要在代码中的finally子句中显式释放锁lock.unlock()。

  另外,在并发量比较小的情况下,使用synchronized是个不错的选择;但是在并发量比较高的情况下,其性能下降会很严重,此时ReentrantLock是个不错的方案。

      在使用synchronized 代码块时,可以与wait()、notify()、nitifyAll()一起使用,从而进一步实现线程的通信。
其中,wait()方法会释放占有的对象锁,当前线程进入等待池,释放cpu,而其他正在等待的线程即可抢占此锁,获得锁的线程即可运行程序;线程的sleep()方法则表示,当前线程会休眠一段时间,休眠期间,会暂时释放cpu,但并不释放对象锁,也就是说,在休眠期间,其他线程依然无法进入被同步保护的代码内部,当前线程休眠结束时,会重新获得cpu执行权,从而执行被同步保护的代码。
wait()和sleep()最大的不同在于wait()会释放对象锁,而sleep()不会释放对象锁。

        notify()方法会唤醒因为调用对象的wait()而处于等待状态的线程,从而使得该线程有机会获取对象锁。调用notify()后,当前线程并不会立即释放锁,而是继续执行当前代码,直到synchronized中的代码全部执行完毕,才会释放对象锁。JVM会在等待的线程中调度一个线程去获得对象锁,执行代码。

  需要注意的是,wait()和notify()必须在synchronized代码块中调用。

  notifyAll()是唤醒所有等待的线程。

 

参考地址:https://www.cnblogs.com/revel171226/p/9411131.html

参考地址:https://www.cnblogs.com/lizhangyong/p/8029287.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值