时间:2021-05-20
重入锁
重入锁,顾名思义,就是支持重进入的锁,它表示该锁能够支持一个线程对资源的重复加锁。重进入是指任意线程在获取到锁之后能够再次获取该锁而不会被锁阻塞,该特性的实现需要解决以下两个问题。
1、线程再次获取锁。锁需要去识别获取锁的线程是否为当前占据锁的线程,如果是,则再次成功获取。
2、锁的最终释放。线程重复n次获取了锁,随后在第n次释放该锁后,其他线程能够获取到该锁。锁的最终释放要求锁对于获取进行计数自增,计数表示当前锁被重复获取的次数,而锁被释放时,计数自减,当计数等于0时表示锁已经成功释放。
Java里面内置锁(synchronize)和Lock(ReentrantLock)都是可重入的
synchronized 实例
package com.home;public class SynchronizedTest implements Runnable { public synchronized void method1() { System.out.println("method1获得锁,正常运行!"); method2(); } public synchronized void method2() { System.out.println("method2获得锁,也正常运行!"); } @Override public void run() { method1(); } public static void main(String[] args) { SynchronizedTest st = new SynchronizedTest(); new Thread(st).start(); new Thread(st).start(); }}Lock 实例
package com.home;import java.util.concurrent.locks.Lock;import java.util.concurrent.locks.ReentrantLock;public class LockTest implements Runnable { Lock lock = new ReentrantLock(); public void method1() { lock.lock(); System.out.println("method1获得锁,正常运行!"); method2(); lock.unlock(); } public void method2() { lock.lock(); System.out.println("method2获得锁,也正常运行!"); lock.unlock(); } @Override public void run() { method1(); } public static void main(String[] args) { LockTest lt = new LockTest(); new Thread(lt).start(); new Thread(lt).start(); }}两个例子最后的结果都是正确的,结果如下:
method1获得锁,正常运行!method2获得锁,也正常运行!method1获得锁,正常运行!method2获得锁,也正常运行!可重入锁最大的作用是避免死锁
读写锁
读写锁维护了一对相关的锁,一个用于只读操作,一个用于写入操作。只要没有writer,读取锁可以由多个reader线程同时保持。写入锁是独占的。
可重入读写锁 ReentrantReadWriteLock
ReentrantReadWriteLock对象提供了readLock()和writeLock()方法, 用于获取读取锁和写入锁.
读取锁允许多个reader线程同时持有, 而写入锁最多只能有一个writter线程持有.
读写锁的使用场合: 读取共享数据的频率远大于修改共享数据的频率. 在上述场合下, 使用读写锁控制共享资源的访问, 可以提高并发性能.
如果一个线程已经持有了写入锁, 则可以再持有读写锁. 相反, 如果一个线程已经持有了读取锁, 则在释放该读取锁之前, 不能再持有写入锁.
可以调用写入锁的newCondition()方法获取与该写入锁绑定的Condition对象, 此时与普通的互斥锁并没有什么区别. 但是调用读取锁的newCondition()方法将抛出异常.
例子
package com.home;import java.util.Random;import java.util.concurrent.locks.ReadWriteLock;import java.util.concurrent.locks.ReentrantReadWriteLock;class ReadWrte { // 共享数据,可以多个线程读数据,只能有一个线程写数据 private int data; // 创建读写锁 ReadWriteLock rwLock = new ReentrantReadWriteLock(); /** * 读数据,上读锁 */ public void get() { // 读锁 rwLock.readLock().lock(); try { System.out.println(Thread.currentThread().getName() + ",Read!"); Thread.sleep((long) Math.random() * 1000); System.out.println(Thread.currentThread().getName() + " 读出的数据为:" + this.getData()); } catch (Exception e) { e.printStackTrace(); } finally { rwLock.readLock().unlock(); } } /** * 写数据,上写锁 * * @param data */ public void put(int data) { // 写锁 rwLock.writeLock().lock(); try { System.out.println(Thread.currentThread().getName() + ",Write!"); Thread.sleep((long) Math.random() * 1000); this.setData(data); System.out.println(Thread.currentThread().getName() + " 写入的数据为:" + this.getData()); } catch (InterruptedException e) { e.printStackTrace(); } finally { rwLock.writeLock().unlock(); } } public int getData() { return data; } public void setData(int data) { this.data = data; }}/** * 测试类 * * @author itmyhome * */public class ReadWriteLockTest { /** * @param args */ public static void main(String[] args) { // 创建ReadWrte对象 final ReadWrte rw = new ReadWrte(); for (int i = 0; i < 10; i++) { // 创建并启动10个读线程 new Thread(new Runnable() { @Override public void run() { rw.get(); } }).start(); // 创建并启动10个写线程 new Thread(new Runnable() { @Override public void run() { // 写入一个随机数 rw.put(new Random().nextInt(8)); } }).start(); } }}输出为
Thread-0,Read!Thread-4,Read!Thread-8,Read!Thread-12,Read!Thread-0 读出的数据为:0Thread-4 读出的数据为:0Thread-8 读出的数据为:0Thread-12 读出的数据为:0Thread-19,Write!Thread-19 写入的数据为:5Thread-7,Write!Thread-7 写入的数据为:7Thread-3,Write!Thread-3 写入的数据为:4Thread-16,Read!Thread-16 读出的数据为:4Thread-11,Write!Thread-11 写入的数据为:0Thread-15,Write!Thread-15 写入的数据为:5Thread-2,Read!Thread-2 读出的数据为:5Thread-17,Write!Thread-17 写入的数据为:2Thread-6,Read!Thread-6 读出的数据为:2Thread-1,Write!Thread-1 写入的数据为:5Thread-13,Write!Thread-13 写入的数据为:4Thread-9,Write!Thread-9 写入的数据为:7Thread-5,Write!Thread-5 写入的数据为:2Thread-10,Read!Thread-10 读出的数据为:2Thread-18,Read!Thread-14,Read!Thread-18 读出的数据为:2Thread-14 读出的数据为:2从图中我们可以看出,可以多个线程同时读,但只能一个线程写,即写数据和写入数据一并完成。
总结
以上就是本文关于Java并发编程之重入锁与读写锁的全部内容,希望对大家有所帮助。欢迎各位参阅本站其他专题,有什么问题可以随时留言,小编会及时回复大家的。感谢大家对本站的支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
Java并发包的locks包里的锁基本上已经介绍得差不多了,ReentrantLock重入锁是个关键,在清楚的了解了同步器AQS的运行机制后,实际上再分析这些锁
在Java并发编程的世界里,synchronized和Lock是控制多线程并发环境下对共享资源同步访问的两大手段。其中Lock是JDK层面的锁机制,是轻量级锁,
问题(1)重入锁是什么?(2)ReentrantLock如何实现重入锁?(3)ReentrantLock为什么默认是非公平模式?(4)ReentrantLock
本文实例讲述了java乐观锁原理与实现。分享给大家供大家参考,具体如下:简单说说乐观锁。乐观锁是相对于悲观锁而言。悲观锁认为,这个线程,发生并发的可能性极大,线
简单回顾一下CAS算法CAS算法即compareandswap(比较与交换),是一种有名的无锁算法。无锁编程,即不使用锁的情况下实现多线程之间的变量同步,也就是