博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java 独占锁|共享锁|互斥锁
阅读量:2455 次
发布时间:2019-05-10

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

独占锁

又叫写锁,指该锁只能被一个线程所持有。Java 中 ReentrantLock 和 synchronized 都是独占锁。

共享锁

又叫读锁,指该锁可以被多个线程所持有。Java 中 ReentrantReadWriteLock 其读锁是共享锁,其写锁是独占锁。

读锁的共享锁可保证并发读是非常高效的,读写,写读,写写的过程是互斥的。

互斥锁

在访问共享资源之前对进行加锁操作,在访问完成之后进行解锁操作。 加锁后,任何其他试图再次加锁的线程会被阻塞,直到当前进程解锁。

加锁前

代码

import java.util.HashMap;import java.util.Map;// 模拟缓存操作class MyChahe{	private volatile Map
map = new HashMap<>(); public void put(String key, Object value) { System.out.println(Thread.currentThread().getName()+"\t 准备写... "+key+" "+value); try { // 停一会 Thread.sleep(300); } catch (InterruptedException e) { e.printStackTrace(); } map.put(key, value); System.out.println(Thread.currentThread().getName()+"\t 写完成..."); } public Object get(String key) { System.out.println(Thread.currentThread().getName()+"\t 准备读..."); try { // 停一会 Thread.sleep(300); } catch (InterruptedException e) { e.printStackTrace(); } Object object = map.get(key); System.out.println(Thread.currentThread().getName()+"\t 读完成... "+object); return object; }}public class ReadWriteLockDemo { public static void main(String[] args) { MyChahe myChahe = new MyChahe(); for (int i = 1; i <= 3; i++) { final int tmpInt = i; new Thread(()->{ myChahe.put(tmpInt+"", tmpInt+""); },String.valueOf(i)).start(); } for (int i = 1; i <= 3; i++) { final int tmpInt = i; new Thread(()->{ myChahe.get(tmpInt+""); },String.valueOf(i)).start(); } }}

运行结果

从运行结果中可以看出,1线程再写入的时候被 2 线程打断,而 2 线程再写入的时候被 3 线程也打断了,最终导致都写入失败了。正常的执行结果应该是 写入->写入成功,中间不允许打断,于是我们给上面的程序进行加锁操作。

加锁后

代码

import java.util.HashMap;import java.util.Map;import java.util.concurrent.locks.ReentrantReadWriteLock;class MyChahe{	private volatile Map
map = new HashMap<>(); // 读写锁 private ReentrantReadWriteLock rwlock = new ReentrantReadWriteLock(); public void put(String key, Object value) { try { rwlock.writeLock().lock(); System.out.println(Thread.currentThread().getName()+"\t 准备写... "+key+" "+value); try { // 停一会 Thread.sleep(300); } catch (InterruptedException e) { e.printStackTrace(); } map.put(key, value); System.out.println(Thread.currentThread().getName()+"\t 写完成..."); } catch (Exception e) { System.out.println(e); }finally { rwlock.writeLock().unlock(); } } public Object get(String key) { try { rwlock.readLock().lock(); System.out.println(Thread.currentThread().getName()+"\t 准备读..."); try { // 停一会 Thread.sleep(300); } catch (InterruptedException e) { e.printStackTrace(); } Object object = map.get(key); System.out.println(Thread.currentThread().getName()+"\t 读完成... "+object); return object; } catch (Exception e) { System.out.println(e); }finally { rwlock.readLock().unlock(); } return null; }}public class ReadWriteLockDemo { public static void main(String[] args) { MyChahe myChahe = new MyChahe(); for (int i = 1; i <= 3; i++) { final int tmpInt = i; new Thread(()->{ myChahe.put(tmpInt+"", tmpInt+""); },String.valueOf(i)).start(); } for (int i = 1; i <= 3; i++) { final int tmpInt = i; new Thread(()->{ myChahe.get(tmpInt+""); },String.valueOf(i)).start(); } }}

运行结果

加锁后运行的结果一致了,写操作不可以被其他线程打断,读操作可以同时多个线程同时读取。

转载地址:http://xlbhb.baihongyu.com/

你可能感兴趣的文章
盲打每分钟资源10几个字_每个系统管理员应了解的10个资源
查看>>
横向扩展基础架构_您应该使用的7种基础架构性能和扩展工具
查看>>
bbc 王超_BBC Microbit入门
查看>>
sysadmin默认密码_Sysadmin感谢日的礼物想法
查看>>
如何在Kubernetes上找到您的Jenkins管理员密码
查看>>
ansible操作数据库_以数据为中心的Ansible修补系统方法
查看>>
c++编写音乐播放器_为什么此开发人员编写了快速响应的音乐播放器
查看>>
github pages_使用此HTTP hack重定向GitHub Pages网站
查看>>
python tox_使用tox自动化Python代码测试
查看>>
python cython_使用Cython为Python编写更快的C扩展
查看>>
flake8变量未使用_使用flake8确保Python代码的一致性
查看>>
ssh与gpg区别_如何使用GPG密钥启用SSH访问进行身份验证
查看>>
apm 韩国开源项目_韩国的开源状态
查看>>
mac上将视频变小_在Mac上将Python 3设置为默认的正确和错误方法
查看>>
java jnlp_Java SE 11删除JNLP的更好解决方案
查看>>
devops 中台_DevOps中的门控生产
查看>>
keil 开源替代_您需要替代开源的哪些专有工具?
查看>>
总论点和分论点_将破坏性的论点变成富有成效的对话
查看>>
pythonic_使用Pythonic在Python中以图形方式编程
查看>>
python black_格式化Python,但您喜欢使用Black
查看>>