1,介绍:

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
 
在JAVA的多线程编程中,我们可以使用synchronized关键字来实现线程之间的同步互斥,但是JDK1.5中新增了ReentrantLock类同样也能达到效果,并且功能上更加强大。比如有嗅探锁定功能,多路分支通知功能,并且使用上比synchronized更加灵活。

2,基本使用:

使用lock()方法上锁,使用unlock()方法解锁
效果:
  • 加锁和解锁之间的代码块,多个线程运行变串行

代码示例:

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /**
* @ClassName LockReentrantLockExample
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/4/24.
*/
public class LockReentrantLockExample {
public static void main(String[] args) {
Lock lock = new ReentrantLock();
Runnable r = () -> {
String tName = Thread.currentThread().getName();
System.out.println("子线程运行");
lock.lock();
System.out.println(tName + "准备睡觉了");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(tName + "睡醒了");
lock.unlock();
}; Thread thread = new Thread(r,"child1");
Thread thread1 = new Thread(r,"child2");
Thread thread2 = new Thread(r,"child3");
thread.start();
thread1.start();
thread2.start();
}
}

3、等待/通知 机制的实现:

3.1 Conditiond对象的await方法实现等待,signal方法实现通知
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /**
* @ClassName LockReentrantLockCondition
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/4/25.
*/
public class LockReentrantLockCondition {
public static void main(String[] args) {
//锁
Lock lock = new ReentrantLock();
//暂停唤醒工具
Condition condition = lock.newCondition();
//子线程本地变量
ThreadLocal<String> tName = new ThreadLocal<>(){
@Override
protected String initialValue() {
return Thread.currentThread().getName();
}
};
Runnable r = new Runnable() {
@Override
public void run() {
System.out.println(tName.get() + "开始运行!");
lock.lock();
try {
Thread.sleep(1000);
System.out.println(tName.get() + "暂停");
condition.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(tName.get() + "运行结束");
lock.unlock();
}
};
Thread thread = new Thread(r,"A");
Thread thread1 = new Thread(r,"B");
Thread thread2 = new Thread(r,"C");
Thread call = new Thread(){
@Override
public void run() {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
lock.lock();
condition.signalAll();
lock.unlock();
}
};
thread.start();
thread1.start();
thread2.start();
call.start(); }
}

效果:锁依次被ABC线程获取到,分别进入await()状态。最后call线程才获取到ABC抛出的锁,

3.2等待通知案例:做馒头,吃馒头,相对优化版

  • 做馒头多线程几乎并发,做馒头耗时并发sleep;
  • 吃馒头几乎并发,sleep 并发执行 lock.unlock();在 sleep外
import java.util.LinkedList;
import java.util.Random;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /**
* @ClassName ReentrantLockConditionMantou
* @projectName: object1
* @author: Zhangmingda
* @description: XXX
* date: 2021/4/25.
*/
public class ReentrantLockConditionMantou { private static void reentrantLockCondition(){
LinkedList<String> mantous = new LinkedList<>();
Lock lock = new ReentrantLock();
Condition fuwuyuan = lock.newCondition();
Runnable produce = () -> {
String threadName = Thread.currentThread().getName();
while (true) {
lock.lock();
if (mantous.size() >= 10 ){
fuwuyuan.signalAll(); //有可能被另一个厨师实例获取到
try {
Thread.sleep(100); //两个厨师,避免厨师之间相互获取到锁,给点时间让食客获取到锁
System.out.println(threadName + ": 馒头做够了.........................你们来吃吧");
fuwuyuan.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
lock.unlock();
}
else {
mantous.add("馒头");
System.out.println(threadName + ",我生产了一个馒头,当前的馒头数量是" + mantous.size());
lock.unlock();
try {
Thread.sleep(1000); //做馒头花时间
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
};
Runnable customer = () -> {
String threadName = Thread.currentThread().getName();
Random random = new Random();
while (true) {
//吃馒头的数量
int buyMantou = Math.abs(random.nextInt()) % 5 + 1;
//获取锁
lock.lock();
if (mantous.size() < buyMantou) {
fuwuyuan.signalAll();
System.out.println("\033[31;1m" + threadName + "想吃" + buyMantou + "只有" +mantous.size() + "个了,快做馒头吧\033[0m");
try {
//稍微歇一歇
Thread.sleep(300); //给点机会上厨师获取到锁,避免其他食客获取到锁
fuwuyuan.await();
lock.unlock();
} catch (InterruptedException e) {
e.printStackTrace();
} } else {
//吃馒头
for (int i = 0; i < buyMantou; i++) {
if(mantous.poll() == null){
System.out.println(threadName + "吃了个空.....");
}
}
System.out.println(threadName + ",我吃了" + buyMantou + "个馒头,当前馒头的数量是:" + mantous.size());
lock.unlock();
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
};
Thread thread = new Thread(produce,"厨师1");
Thread thread1 = new Thread(produce,"厨师2");
Thread thread2 = new Thread(customer,"饭桶1");
Thread thread3 = new Thread(customer,"饭桶2");
Thread thread4 = new Thread(customer,"饭桶3");
thread.start();
thread1.start();
thread2.start();
thread3.start();
thread4.start();
}
public static void main(String[] args) {
reentrantLockCondition();
}
}

"C:\Program Files\JetBrains\IntelliJ IDEA Community Edition 2019.3\jbr\bin\java.exe" "-javaagent:C:\Program Files\JetBrains\IntelliJ IDEA Community Edition 2019.3\lib\idea_rt.jar=54462:C:\Program Files\JetBrains\IntelliJ IDEA Community Edition 2019.3\bin" -Dfile.encoding=UTF-8 -classpath D:\JavaStudy2\object1\out\production\多线程 ReentrantLockConditionMantou
厨师1,我生产了一个馒头,当前的馒头数量是1
厨师2,我生产了一个馒头,当前的馒头数量是2
饭桶3,我吃了2个馒头,当前馒头的数量是:0
饭桶1想吃4只有0个了,快做馒头吧
饭桶2想吃3只有0个了,快做馒头吧
饭桶1想吃5只有0个了,快做馒头吧
饭桶2想吃4只有0个了,快做馒头吧
厨师1,我生产了一个馒头,当前的馒头数量是1
厨师2,我生产了一个馒头,当前的馒头数量是2
饭桶1想吃4只有2个了,快做馒头吧
饭桶2想吃5只有2个了,快做馒头吧
饭桶1,我吃了1个馒头,当前馒头的数量是:1
饭桶3,我吃了1个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
厨师1,我生产了一个馒头,当前的馒头数量是3
厨师2,我生产了一个馒头,当前的馒头数量是4
饭桶1,我吃了3个馒头,当前馒头的数量是:1
饭桶3想吃3只有1个了,快做馒头吧
饭桶2想吃3只有1个了,快做馒头吧
厨师1,我生产了一个馒头,当前的馒头数量是2
厨师2,我生产了一个馒头,当前的馒头数量是3
饭桶3想吃5只有3个了,快做馒头吧
饭桶2想吃4只有3个了,快做馒头吧
饭桶3,我吃了1个馒头,当前馒头的数量是:2
厨师1,我生产了一个馒头,当前的馒头数量是3
厨师2,我生产了一个馒头,当前的馒头数量是4
饭桶1,我吃了2个馒头,当前馒头的数量是:2
厨师1,我生产了一个馒头,当前的馒头数量是3
厨师2,我生产了一个馒头,当前的馒头数量是4
饭桶3想吃5只有4个了,快做馒头吧
饭桶2,我吃了4个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶1想吃4只有2个了,快做馒头吧
饭桶3想吃5只有2个了,快做馒头吧
饭桶1,我吃了2个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶2想吃5只有2个了,快做馒头吧
饭桶3想吃4只有2个了,快做馒头吧
厨师2,我生产了一个馒头,当前的馒头数量是3
厨师1,我生产了一个馒头,当前的馒头数量是4
饭桶2,我吃了1个馒头,当前馒头的数量是:3
饭桶1,我吃了3个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶2想吃5只有2个了,快做馒头吧
饭桶3想吃3只有2个了,快做馒头吧
厨师1,我生产了一个馒头,当前的馒头数量是3
厨师2,我生产了一个馒头,当前的馒头数量是4
饭桶2,我吃了1个馒头,当前馒头的数量是:3
饭桶1,我吃了3个馒头,当前馒头的数量是:0
厨师1,我生产了一个馒头,当前的馒头数量是1
厨师2,我生产了一个馒头,当前的馒头数量是2
厨师2,我生产了一个馒头,当前的馒头数量是3
饭桶2,我吃了1个馒头,当前馒头的数量是:2
厨师1,我生产了一个馒头,当前的馒头数量是3
饭桶1想吃5只有3个了,快做馒头吧
饭桶3,我吃了1个馒头,当前馒头的数量是:2
厨师1,我生产了一个馒头,当前的馒头数量是3
厨师2,我生产了一个馒头,当前的馒头数量是4
饭桶2,我吃了3个馒头,当前馒头的数量是:1
厨师2,我生产了一个馒头,当前的馒头数量是2
厨师1,我生产了一个馒头,当前的馒头数量是3
饭桶3,我吃了2个馒头,当前馒头的数量是:1
厨师1,我生产了一个馒头,当前的馒头数量是2
厨师2,我生产了一个馒头,当前的馒头数量是3
饭桶2,我吃了2个馒头,当前馒头的数量是:1
厨师1,我生产了一个馒头,当前的馒头数量是2
厨师2,我生产了一个馒头,当前的馒头数量是3
饭桶3想吃5只有3个了,快做馒头吧
饭桶1想吃5只有3个了,快做馒头吧
饭桶3,我吃了2个馒头,当前馒头的数量是:1
厨师1,我生产了一个馒头,当前的馒头数量是2
厨师2,我生产了一个馒头,当前的馒头数量是3
饭桶2想吃4只有3个了,快做馒头吧
饭桶1,我吃了3个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶3想吃5只有2个了,快做馒头吧
饭桶2,我吃了1个馒头,当前馒头的数量是:1
厨师1,我生产了一个馒头,当前的馒头数量是2
厨师2,我生产了一个馒头,当前的馒头数量是3
厨师1,我生产了一个馒头,当前的馒头数量是4
厨师2,我生产了一个馒头,当前的馒头数量是5
饭桶1,我吃了5个馒头,当前馒头的数量是:0
饭桶2想吃4只有0个了,快做馒头吧
饭桶3想吃3只有0个了,快做馒头吧
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶2,我吃了2个馒头,当前馒头的数量是:0
饭桶1想吃1只有0个了,快做馒头吧
饭桶3想吃2只有0个了,快做馒头吧
饭桶1想吃1只有0个了,快做馒头吧
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶3,我吃了1个馒头,当前馒头的数量是:1
饭桶2,我吃了1个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶3想吃4只有2个了,快做馒头吧
饭桶1,我吃了2个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶2想吃3只有2个了,快做馒头吧
饭桶3想吃3只有2个了,快做馒头吧
饭桶2,我吃了2个馒头,当前馒头的数量是:0
厨师2,我生产了一个馒头,当前的馒头数量是1
厨师1,我生产了一个馒头,当前的馒头数量是2
饭桶1想吃5只有2个了,快做馒头吧
饭桶3想吃5只有2个了,快做馒头吧
厨师2,我生产了一个馒头,当前的馒头数量是3
厨师1,我生产了一个馒头,当前的馒头数量是4
饭桶1想吃5只有4个了,快做馒头吧
饭桶3,我吃了3个馒头,当前馒头的数量是:1
饭桶2想吃5只有1个了,快做馒头吧
饭桶1想吃3只有1个了,快做馒头吧 Process finished with exit code -1

测试输出

公平锁和非公平锁:

锁Lock分为公平锁和非公平锁,公平锁表示线程获取锁的顺序是按照线程加锁的顺序来分配的,也就是先进先出的顺序。而非公平锁就是一种获取锁的抢占机制,是随机获取的,和公平锁不一样的就是先来的不一定先得到锁,这个方式可能造成某些线程一直拿不到锁。如果我们用synchronized的方式我们没有办法使用公平锁。
创建公平锁的办法是在new ReentrantLock的时候加一个参数true.
new ReentrantLock(true)

java 多线程 Thread 锁ReentrantLock;Condition等待与通知;公平锁的更多相关文章

  1. java多线程20 : ReentrantLock中的方法 ,公平锁和非公平锁

    公平锁与非公平锁 ReentrantLock有一个很大的特点,就是可以指定锁是公平锁还是非公平锁,公平锁表示线程获取锁的顺序是按照线程排队的顺序来分配的,而非公平锁就是一种获取锁的抢占机制,是随机获得 ...

  2. “全栈2019”Java多线程第三十二章:显式锁Lock等待唤醒机制详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  3. java并发编程——通过ReentrantLock,Condition实现银行存取款

         java.util.concurrent.locks包为锁和等待条件提供一个框架的接口和类,它不同于内置同步和监视器.该框架允许更灵活地使用锁和条件,但以更难用的语法为代价. Lock 接口 ...

  4. java多线程6:ReentrantLock

    下面看下JUC包下的一大并发神器ReentrantLock,是一个可重入的互斥锁,具有比synchronized更为强大的功能. ReentrantLock基本用法 先来看一下ReentrantLoc ...

  5. Java多线程学习(四)等待/通知(wait/notify)机制

    转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79690279 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...

  6. Java多线程(五)——线程等待与唤醒

    一.wait().notify().notifyAll()等方法介绍 在Object.java中,定义了wait(), notify()和notifyAll()等接口.wait()的作用是让当前线程进 ...

  7. “全栈2019”Java多线程第二十九章:可重入锁与不可重入锁详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  8. “全栈2019”Java多线程第七章:等待线程死亡join()方法详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  9. 【分布式锁】02-使用Redisson实现公平锁原理

    前言 前面分析了Redisson可重入锁的原理,主要是通过lua脚本加锁及设置过期时间来保证锁执行的原子性,然后每个线程获取锁会将获取锁的次数+1,释放锁会将当前锁次数-1,如果为0则表示释放锁成功. ...

随机推荐

  1. Go语言核心36讲(Go语言实战与应用二十)--学习笔记

    42 | bufio包中的数据类型 (上) 今天,我们来讲另一个与 I/O 操作强相关的代码包bufio.bufio是"buffered I/O"的缩写.顾名思义,这个代码包中的程 ...

  2. char和varchar2

    1.CHAR的长度是固定的,而VARCHAR2的长度是可以变化的. 比如,存储字符串"abc",对于CHAR (10),表示你存储的字符将占10个字节(包括7个空字符) 而同样的V ...

  3. Notepad++—显示代码对齐是使用了制表符还是空格

    使用Notepad++打开脚本,勾选"显示空格与制表符",此时你会看到代码对齐使用了制表符与空格 右箭头:TAB:空格:点: 参考:https://www.cnblogs.com/ ...

  4. python函数初体验

    函数 函数参数w 形式参数>>>>(被指定具体的值)默认参数, 实际参数是调用时候的实际指定参数 我们把函数⾥⾯的参数叫形式函数,函数实际调⽤的时候,赋予的参数叫实际函数 定义 ...

  5. TD课程通最终版本体验

    功能上,新版本增加了学校教室的上课情况,有无课程可以清楚查询,如下图: 在添加课程的设置上有改进,相比于之前编辑课程后不能保存,新版本在可保存的基础上又增加了登陆教务系统的功能,学生使用更加方便快捷, ...

  6. 用python写的推箱子搜索程序

    1 # -*- coding: gbk -*- 2 from functools import reduce 3 from copy import deepcopy 4 import re 5 def ...

  7. Angular中怎样创建service服务来实现组件之间调用公共方法

    Angular组件之间不能互相调用方法,但是可以通过创建服务来实现公共方法的调用. 实现 创建服务命令 ng g service 服务路径/服务名 比如这里在app/services目录下创建stor ...

  8. 【php安全】 register_argc_argv 造成的漏洞分析

    对register_argc_argv的分析 简介 使用 cli模式下,不论是否开始register_argc_argv,都可以获取命令行或者说外部参数 web模式下,只有开启了register_ar ...

  9. 如何通过 User-Agent 识别百度蜘蛛

    如果有大量的百度蜘蛛抓取网站就需要注意了:有可能是其他爬虫伪造百度蜘蛛恶意抓取网站. 如果遇到这种情况,这时候就需要查看日志来确定是不是真正的百度蜘蛛(baidu spider).搜索引擎蜘蛛.用户访 ...

  10. shell神器curl命令的用法 curl用法实例笔记

    shell神器curl命令的用法举例,如下: ##基本用法(配合sed/awk/grep) $curl http://www.jquerycn.cn ##下载保存 $curl http://www.j ...