之前在写显示锁的是后,在显示锁的接口中,提到了new Condition这个方法,这个方法会返回一个Condition对象

简单介绍一下

Condition接口:

  任意一个Java对象,都拥有一组监视器方法(定义在java.lang.Object上),主要包括wait()、wait(long timeout)、notify()以及notifyAll()方法,这些方法与synchronized同步关键字配合,可以实现等待/通知模式。之前写过一篇线程之间的协作(等待通知模式)是使用Object的wait和notify/notifyAll+Synchronized写的

  换而言之,synchronized关键字想要实现等待/通知模式,需要调用以上的四种方法。

  然后我们的Condition接口也提供了能够实现等待/通知模式,是与Lock配合实现的。

  我感觉这个Condition和那个差不多,也是用来完成线程之间的协作的

  但是二者在使用方式上以及功能特性上还是有所差别的。

Object对比Condition:

由此表可以看出,condition接口可以有多个等待队列,而object监视器方法只有一个队列,而且还不支持在等待状态响应中断,还不支持当前线程释放锁并进入等待状态到将来的某个时间。

示例:

  也不打算写新的示例了,用这个Condition接口改造一下之前使用等待通知模式的那个案例吧

  Condition定义了等待/通知两种类型的方法,当前线程调用这些方法时,需要提前获取到Condition对象关联的锁。Condition对象是由Lock对象(调用Lock对象的newCondition()方法)创建出来的。其实就是,Condition是依赖Lock对象的。就像使用wait/notify需要依赖Synchronized锁一样,Condition的使用方式比较简单,需要注意在调用方法前获取锁

创建等待通知类

package org.dance.day4.condition;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /**
* 类说明:使用Condition接口实现等待通知模式
*/
public class ExpressCond { public final static String CITY = "ShangHai"; /**
* 快递运输里程数
*/
private int km; /**
* 快递到达地点
*/
private String site; /**
* 创建显示锁
*/
private final Lock lock = new ReentrantLock(); /**
* 检测城市变化
*/
private final Condition siteCond = lock.newCondition(); /**
* 检测公里数变化
*/
private final Condition kmCond = lock.newCondition(); public ExpressCond() {
} public ExpressCond(int km, String site) {
this.km = km;
this.site = site;
} /* 变化公里数,然后通知处于wait状态并需要处理公里数的线程进行业务处理*/
public void changeKm() {
// 获取锁
lock.lock();
try {
this.km = 101;
// 唤醒在kmCond 上 等待的线程
kmCond.signal();
} finally {
lock.unlock();
}
} /* 变化地点,然后通知处于wait状态并需要处理地点的线程进行业务处理*/
public void changeSite() {
// 获取锁
lock.lock();
try {
this.site = "BeiJing";
// 唤醒在siteCond 上 等待的线程
siteCond.signal();
} finally {
lock.unlock();
}
} /*当快递的里程数大于100时更新数据库*/
public void waitKm() {
lock.lock();
try {
while (this.km <= 100) {
try {
kmCond.await();
} catch (InterruptedException e) {
// 处理线程中断
Thread.currentThread().interrupt();
e.printStackTrace();
}
System.out.println("check km thread[" + Thread.currentThread().getId()
+ "] is be notifed.");
}
} finally {
lock.unlock();
}
System.out.println("the Km is " + this.km + ",I will change db");
} /*当快递到达目的地时通知用户*/
public void waitSite() {
lock.lock();
try {
while (CITY.equals(this.site)) {
try {
siteCond.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("check site thread[" + Thread.currentThread().getId()
+ "] is be notifed.");
}
} finally {
lock.unlock();
}
System.out.println("the site is " + this.site + ",I will call user");
}
}

通过代码可以看见,我们一个锁,是可以携带多个等待队列的

创建测试类

package org.dance.day4.condition;

/**
*类说明:测试Lock和Condition实现等待通知
*/
public class TestCond {
private static ExpressCond express = new ExpressCond(0,ExpressCond.CITY); /*检查里程数变化的线程,不满足条件,线程一直等待*/
private static class CheckKm extends Thread{
@Override
public void run() {
express.waitKm();
}
} /*检查地点变化的线程,不满足条件,线程一直等待*/
private static class CheckSite extends Thread{
@Override
public void run() {
express.waitSite();
}
} public static void main(String[] args) throws InterruptedException {
for(int i=0;i<3;i++){
new CheckSite().start();
}
for(int i=0;i<3;i++){
new CheckKm().start();
} Thread.sleep(1000);
express.changeKm();//快递里程变化
}
}

执行结果:

check km thread[14] is be notifed.
the Km is 101,I will change db

通过执行结果,我们可以清晰的看到,他是直接唤醒了,在公里数变化上等待的线程的,在之前的等待通知模式中,也就是wait/notify/notifyAll+Sync实现的等待通知模式中,推荐大家使用notifyAll()来唤醒正在等待中的线程,但是在使用Condition接口中,推荐大家使用signal,而不是signalAll().为啥呢?因为wait/notify/notifyAll是Object的方法,在指定的对象中等待的可能是多个线程,分别在检测不同的变量,可能造成信号的拦截,所以推荐使用全部唤醒,,但是在使用Condition上却不是,因为他是多个等待队列,他清晰的知道自己应该唤醒那个线程,所以推荐使用signal,至于Condition的实现分析暂时先不写,等写完AQS再写方便,大家理解,我感觉不是所有相关的知识都要堆在一起,要是理解不了,再深入也没用

作者:彼岸舞

时间:2020\11\04

内容关于:并发编程

本文来源于网络,只做技术分享,一概不负任何责任

Lock接口之Condition接口的更多相关文章

  1. jdk1.5多线程Lock接口及Condition接口

    jdk1.5多线程的实现的方式: jdk1.5之前对锁的操作是隐式的 synchronized(对象) //获取锁 { } //释放锁 jdk1.5锁的操作是显示的:在包java.util.concu ...

  2. java中的锁之Lock接口与Condition接口

    一.Lock源码. 1.是一个接口.一共有6个方法. 2.方法详细如下: (1)当前线程尝试获取锁.结果分两种情况,一是成功获取到锁,则返回:二是获取锁失败,则一直等待.不响应中断请求. (2)当前线 ...

  3. 并发之lock的condition接口

    13.死磕Java并发-----J.U.C之Condition 12.Condition使用总结 11.Java并发编程系列之十七:Condition接口 === 13.死磕Java并发-----J. ...

  4. Java之多线程开发时多条件Condition接口的使用

    转:http://blog.csdn.net/a352193394/article/details/39454157 我们在多线程开发中,可能会出现这种情况.就是一个线程需要另外一个线程满足某某条件才 ...

  5. 6.类似Object监视器方法的Condition接口

    在<1.有关线程.并发的基本概念>中,我们利用synchronized关键字.Queue队列.以及Object监视器方法实现了生产者消费者,介绍了有关线程的一些基本概念.Object类提供 ...

  6. Java的LockSupport工具,Condition接口和ConditionObject

    在之前我们文章(关于多线程编程基础和同步器),我们就接触到了LockSupport工具和Condition接口,之前使用LockSupport工具来唤醒阻塞的线程,使用Condition接口来实现线程 ...

  7. Condition接口

    <Java并发编程艺术>读书笔记 Condition介绍 任意一个Java对象,都拥有一组监视器方法(定义在java.lang.Object中),主要包括wait().wait(long ...

  8. Java并发Condition接口

    java.util.concurrent.locks.Condition接口提供一个线程挂起执行的能力,直到给定的条件为真. Condition对象必须绑定到Lock,并使用newCondition( ...

  9. Condition接口及其主要实现类ConditionObject源码浅析

    1.引子 任意一个Java对象,都拥有一组监视器方法(定义在java.lang.Object上),主要包括wait().wait(long timeout).notify()以及notifyAll() ...

随机推荐

  1. Centos-退出抽取设备-eject

    eject 退出抽取设备,如光驱或磁带,如果设备已经挂载,则卸载设备 相关选项 -q 退出磁盘 -r 退出光盘 -d 显示默认设备

  2. Optimisation

    https://www.cnblogs.com/wuyudong/p/writing-efficient-c-and-code-optimization.html 1 不要过多使用 stack ,尽量 ...

  3. linux系统或centos7安装nginx

    一.Linux下安装nginx 1.添加源 sudo rpm -Uvh http://nginx.org/packages/centos/7/noarch/RPMS/nginx-release-cen ...

  4. Anaconda安装Pytorch(通过本地安装包)

    前提:你已经事先安装好了Anaconda 在线安装pytorch总是出现这样那样的问题,所以我选择先去清华镜像下载好与python版本对应的pytorch和torchvision文件,然后本地安装 清 ...

  5. Web Storage API的介绍和使用

    目录 简介 浏览器的本地存储技术 Web Storage相关接口 浏览器兼容性 隐身模式 使用Web Storage API 总结 简介 Web Storage为浏览器提供了方便的key value存 ...

  6. 03 sublime text3下配置Java的编译运行环境

    参考如下文章,加入了自己的干货: https://blog.csdn.net/qq_38295511/article/details/81140069 https://blog.csdn.net/qq ...

  7. springcloud学习入门

    Springcloud入门学习笔记 1. 项目初始化配置 1. 1. 新建maven工程 使用idea创建maven项目 1. 2. 在parent项目pom中导入以下依赖 <parent> ...

  8. JDK 中的栈竟然是这样实现的?

    前面的文章<动图演示:手撸堆栈的两种实现方法!>我们用数组和链表来实现了自定义的栈结构,那在 JDK 中官方是如何实现栈的呢?接下来我们一起来看. 这正式开始之前,先给大家再解释一下「堆栈 ...

  9. 两个多维高斯分布之间的KL散度推导

    在深度学习中,我们通常对模型进行抽样并计算与真实样本之间的损失,来估计模型分布与真实分布之间的差异.并且损失可以定义得很简单,比如二范数即可.但是对于已知参数的两个确定分布之间的差异,我们就要通过推导 ...

  10. ansible-介绍

    常用自动化运维工具 CFengine Chef Puppet 基于Ruby开发,采用C/S架构,扩展性强,基于SSL认证 SaltStack 基于python开发,采用C/S架构,相对于puppet更 ...