JUC并发编程-锁分类
2022/1/29 11:34:16
本文主要是介绍JUC并发编程-锁分类,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
锁分类
公平和非公平锁
并发包中ReentrantLock
的创建可以指定构造函数的boolean类型来决定是公平锁还是非公平锁,默认是非公平锁
公平锁
公平锁,在并发环境下,每个线程在获取锁的时候会去查看此锁维护的等待队列,如果为空,或者当前线程为等待队列的第一个,就占有锁,否则加入到等待队列里,然后按照FIFO(先进先出)的规则等待
非公平锁
不公平,上来就直接尝试占有锁,就想插队,如果插不上,就再采用类似公平锁的方式,例如synchronized
可重入锁(又名递归锁)
指同一个线程在获得锁之后,遇到需要相同锁的同步代码,会自动获取锁。
线程可以进入任何一个他已经拥有的锁所同步的代码块
自旋锁
尝试获取锁的线程没拿到锁之后不会立即阻塞,而是采用循环的方式尝试获取锁,可以减少线程上下文切换,消耗cpu性能
public class SpinLockDemo { AtomicReference<Thread> atomicReference=new AtomicReference(); public void myLock(){ //获取当前线程 Thread thread=Thread.currentThread(); while (!atomicReference.compareAndSet(null, thread)){ } System.out.println(Thread.currentThread().getName()+"获得锁"); } public void unLock(){ Thread thread=Thread.currentThread(); atomicReference.compareAndSet(thread,null); System.out.println(Thread.currentThread().getName()+"解锁"); } public static void main(String[] args) throws InterruptedException { SpinLockDemo spinLockDemo=new SpinLockDemo(); new Thread(()->{ spinLockDemo.myLock(); try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } spinLockDemo.unLock(); },"t1").start(); Thread.sleep(1000); new Thread(()->{ spinLockDemo.myLock(); spinLockDemo.unLock(); },"t2").start(); } }
独占锁(写锁)/共享锁(读锁)/互斥锁
-
独占锁:指该锁一次只能被一个线程所持有、对ReentrantLock和Synchronized而言都是独占锁
-
共享锁:指该锁可被多个线程所持有
-
互斥锁:读锁的共享锁可以保证并发读是非常高效的,读写、写读、写写的过程是互斥的
ReentrantReadWriteLock
其读锁是共享锁,写锁是独占锁
为了增加并发性能,读取共享资源的时候可以同时进行,但是写入共享资源的时候,不允许其他线程对资源进行读写
- 读读共存
- 读写互斥
- 写写互斥
public class ReadWriteLockDemo { public static void main(String[] args) { MyCache myCache = new MyCache(); for (int i = 0; i < 5; i++) { //匿名内部类使用外部的局部变量必须用final修饰保证数据的一致性,局部变量和内部类生命周期不一样 final int tem=i; new Thread(()->{ try { myCache.put(tem+"",tem+""); } catch (InterruptedException e) { e.printStackTrace(); } },String.valueOf(i)).start(); } for (int i = 0; i < 5; i++) { //匿名内部类使用外部的局部变量必须用final修饰保证数据的一致性,局部变量和内部类生命周期不一样 final int tem=i; new Thread(()->{ try { myCache.get(tem+""); } catch (InterruptedException e) { e.printStackTrace(); } },String.valueOf(i)).start(); } } } class MyCache{ private volatile Map<String,Object> map=new HashMap<>(); //可重入读写锁 private ReentrantReadWriteLock rwLock=new ReentrantReadWriteLock(); public void put(String key,Object value) throws InterruptedException { //写锁 rwLock.writeLock().lock(); System.out.println(Thread.currentThread().getName()+": 正在写入"); Thread.sleep(1000); map.put(key,value); System.out.println(Thread.currentThread().getName()+": 写入完成"); // rwLock.writeLock().unlock(); } public void get(String key) throws InterruptedException { //读锁 rwLock.readLock().lock(); System.out.println(Thread.currentThread().getName()+": 读取中"); Thread.sleep(1000); System.out.println(Thread.currentThread().getName()+": 读取完成:"+map.get(key)); //解锁 rwLock.readLock().unlock(); }
这篇关于JUC并发编程-锁分类的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2024-11-23增量更新怎么做?-icode9专业技术文章分享
- 2024-11-23压缩包加密方案有哪些?-icode9专业技术文章分享
- 2024-11-23用shell怎么写一个开机时自动同步远程仓库的代码?-icode9专业技术文章分享
- 2024-11-23webman可以同步自己的仓库吗?-icode9专业技术文章分享
- 2024-11-23在 Webman 中怎么判断是否有某命令进程正在运行?-icode9专业技术文章分享
- 2024-11-23如何重置new Swiper?-icode9专业技术文章分享
- 2024-11-23oss直传有什么好处?-icode9专业技术文章分享
- 2024-11-23如何将oss直传封装成一个组件在其他页面调用时都可以使用?-icode9专业技术文章分享
- 2024-11-23怎么使用laravel 11在代码里获取路由列表?-icode9专业技术文章分享
- 2024-11-22怎么实现ansible playbook 备份代码中命名包含时间戳功能?-icode9专业技术文章分享