文档章节

多线程高并发系列二(ReenTrantLock锁可以替代synchronized锁)

字数 736
阅读 8
收藏 0

【推荐】2019 Java 开发者跳槽指南.pdf(吐血整理) >>>

ReenTrantLock可以替代synchronized锁,并且比synchronized锁更灵活

  1. synchronized锁是自动上锁、自动解锁,而ReenTrantLock需要手动上锁、手动解锁
  2. synchronized锁在程序运行时,如果抛异常,jvm会自动释放锁,而ReenTrantLock还是得自己手动释放锁,所以,释放锁一般都是写在finally中
  3. **ReentrantLock的tryLock方法,是尝试获取锁。**就是去尝试获取锁,获取不到就继续往下执行,不想synchronized锁,获取不到锁,就在那死等,该方法有一个boolean类型的返回值,你可以根据这个返回值,执行你的逻辑。并且可以指定尝试获取锁的时间,相当于等待获取锁的时间。
  4. ReentrantLock的lockInterruptibly获取锁,除了tryLock,通过lockInterruptibly方法也可以获取锁,可以对线程的interrupt方法作出响应。这个方法的意义有点类似于tryLock使用超时的时候的场景。:两个线程,T1获取锁,T2线程启动,获取不到锁,然后你不想让T2等了,如果用lock、tryLock是没法打断的,如果用lockInterruptibly是可以打断的。
  5. ReentrantLock可以是一个公平锁,公平锁就是,哪个线程等锁等的时间长,就先执行哪个。而synchronized锁是非公平锁。
public class ReentrantLockTest {
	public static void main(String[] args) {
		Lock rtLock = /*new ReentrantLock(true) 公平锁*/
					new ReentrantLock();
		Thread t1 = new Thread(()->{
			try {
				rtLock.lock();
//				rtLock.tryLock();尝试获取锁
//				rtLock.tryLock(2,TimeUnit.SECONDS);尝试获取锁,超时设置
				TimeUnit.SECONDS.sleep(Integer.MAX_VALUE);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}finally {
				rtLock.unlock();
			}
		});
		t1.start();
		Thread t2 = new Thread(()->{
			try {
//				rtLock.lock();
				rtLock.lockInterruptibly();
				TimeUnit.SECONDS.sleep(2);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}finally {
				rtLock.unlock();
			}
		});
		t2.start();
		try {
			TimeUnit.SECONDS.sleep(2);
			t2.interrupt();
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}
  1. ReentrantLock还可以绑定conditon,看代码你就懂了
public class ConditionTest {
	//使用condition来完成这个操作:有2个生产者线程、10个消费者线程,如果容器满了,则生产者暂停,如果容器空了,则消费者暂停
	final private LinkedList list = new LinkedList();
	final private int maxValue = 10;
	Lock lock = new ReentrantLock();
	private Condition producerCondition = lock.newCondition();//生产者
	private Condition consumerCondition = lock.newCondition();//消费者
	/**
	 * 生产者,按照要求:当容器满了,就停止。
	 */
	public void producer(Object o) {
		while(list.size()==maxValue) {
			try {
//				this.wait(); wait 是配合synchronized锁用的
				producerCondition.await();
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
		list.add(o);
		consumerCondition.signalAll();//只唤醒了消费者线程
	}
	/**
	 * 消费者
	 */
	public  Object consumer() {
		while(list.size()==0) {
			try {
				consumerCondition.await();
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
		}
		Object o = list.removeFirst();
		producerCondition.signalAll();
		return o;
	}
	public static void main(String[] args) {
		ConditionTest conditionTest = new ConditionTest();
		//启动消费者线程
		for(int i=0;i<10;i++) {
			new Thread(()->{
				for(int j=0;j<10;j++) {
					System.out.println(conditionTest.consumer()+"aa");
				}
			},"c"+i) .start();
		}
		try {
			TimeUnit.SECONDS.sleep(2);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		//启动生产者线程
		for(int i=0;i<2;i++) {
			new Thread(()->{
				for(int j=0;j<30;j++) {
					conditionTest.producer(Thread.currentThread().getName());
				}
			},"p"+i) .start();
		}
	}
}

© 著作权归作者所有

粉丝 0
博文 16
码字总数 25998
作品 0
通州
私信 提问
[高并发Java 五] JDK并发包1

在[高并发Java 二] 多线程基础中,我们已经初步提到了基本的线程同步操作。这次要提到的是在并发包中的同步控制工具。 1. 各种同步控制工具的使用 1.1 ReentrantLock ReentrantLock感觉上是s...

Hosee
2016/01/21
11.1K
0
4种常用Java线程锁的特点,性能比较及使用场景

多个线程同时对同一个对象进行读写操作,很容易会出现一些难以预料的问题。所以很多时候我们需要给代码块加锁,同一时刻只允许一个线程对某个对象进行操作。多线程之所以会容易引发一些难以发现...

mikechen优知
03/10
216
0
死磕 java同步系列之ReentrantLock VS synchronized——结果可能跟你想的不一样

问题 (1)ReentrantLock有哪些优点? (2)ReentrantLock有哪些缺点? (3)ReentrantLock是否可以完全替代synchronized? 简介 synchronized是Java原生提供的用于在多线程环境中保证同步的...

彤哥读源码
06/11
2.2K
8
Synchronized与Lock

Synchronized是并发中使用最频繁的关键字了,它可以使用在方法、代码块上,表示对该段代码加锁,而必须要持有锁才能执行这段代码。Synchronized具有互斥性。 说起来蛮简单,而实际中编程 ,最...

EveryDayNew
2016/10/02
80
0
并发编程-锁的发展和主流分布式锁比较总结

一、锁的发展 系统结构由传统的“单应用服务--》SOA --》微服务 --》无服务器” 的演进过程中,场景越来越复杂,由单体应用的但进程中多线程并发的内存锁,随着互联网场景越来越复杂,在复杂...

贾浩v
2017/10/24
297
0

没有更多内容

加载失败,请刷新页面

加载更多

加载JDBC驱动

我们平时在连接数据库时需要加载驱动,通常做法是将JDBC驱动程序放在类路径中的某个位置,然后用Class.forName()查找并加载驱动程序。 这也就意味着要么将驱动程序打包到jar中,要么将驱动程...

uknow8692
22分钟前
4
0
TCP三次握手详情

TCP(Transmission Control Protocol,传输控制协议)是一种面向连接的、可靠的、基于字节流的通信协议,数据在传输前要建立连接,传输完毕后还要断开连接。 客户端在收发数据前要使用 conn...

vinci321
23分钟前
4
0
适用于PDF文件的MIME媒体类型

使用PDF时,我遇到了MIME类型application/pdf和application/x-pdf等。 这两种类型之间是否存在差异,如果是这样,它是什么? 一个比另一个更受欢迎吗? 我正在开发一个必须提供大量PDF的网络...

技术盛宴
30分钟前
5
0
Docker数据挂载

Docker数据管理 在容器中管理数据主要有两种方式: 数据卷(Volumes) 挂载主机目录(Bind mounts) 数据卷 数据卷是一个可供一个或多个容器使用的特殊目录,它绕过UFS,可以提供很多有用的特...

CodingDiary
42分钟前
6
0
jQuery获取特定的选项标签文本

好吧,说我有这个: <select id='list'> <option value='1'>Option A</option> <option value='2'>Option B</option> <option value='3'>Option C</option></select> 如果我想......

javail
46分钟前
4
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部