Java线程同步的Monitor机制(Lock配合Condition)
Monitor模式是一种常见的并行开发机制, 一个Monitor实例可以被多个线程安全使用, 所有的monitor下面的方法在运行时是互斥的, 这种互斥机制机制可以用于一些特性, 例如让线程等待某种条件, 在等待时线程会将CPU时间交出去, 但是在条件满足时确保重新获得CPU时间. 在条件达成时, 你可以同时通知一个或多个线程. 这样做有以下的优点:
- 所有的同步代码都集中在一起, 用户不需要知道这是如何实现的
- 代码不依赖于线程数量, 线程数量只取决于业务需要
- 不需要对某个互斥对象做释放, 不存在忘记的风险
一个Monitor的结构是这样的
public class SimpleMonitor {
public method void testA(){
//Some code
}
public method int testB(){
return 1;
}
}
使用Java代码不能直接创建一个Monitor, 要实现Monitor, 需要使用Lock和Condition类. 一般使用的Lock是ReentrantLock, 例如
public class SimpleMonitor {
private final Lock lock = new ReentrantLock();
public void testA() {
lock.lock();
try {
//Some code
} finally {
lock.unlock();
}
}
public int testB() {
lock.lock();
try {
return 1;
} finally {
lock.unlock();
}
}
}
如果不需要判断条件, 那么用synchronized就可以了. 在需要判断条件的情况下, 使用Lock的newCondition()方法创建Condition, 可以通过Condition的await方法, 让当前线程wait, 放弃cpu时间. 然后用signal或者signalAll方法让线程重新获得CPU时间. signalAll方法会唤起所有wait在当前condition的线程. 下面是一个例子, 一个需要被多个线程使用的容量固定的buffer.
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class BoundedBuffer {
private final String[] buffer;
private final int capacity; private int front;
private int rear;
private int count; private final Lock lock = new ReentrantLock(); private final Condition notFull = lock.newCondition();
private final Condition notEmpty = lock.newCondition(); public BoundedBuffer(int capacity) {
super(); this.capacity = capacity; buffer = new String[capacity];
} public void deposit(String data) throws InterruptedException {
lock.lock(); try {
while (count == capacity) {
notFull.await();
} buffer[rear] = data;
rear = (rear + 1) % capacity;
count++; notEmpty.signal();
} finally {
lock.unlock();
}
} public String fetch() throws InterruptedException {
lock.lock(); try {
while (count == 0) {
notEmpty.await();
} String result = buffer[front];
front = (front + 1) % capacity;
count--; notFull.signal(); return result;
} finally {
lock.unlock();
}
}
}
代码说明
- 这两个方法通过lock互斥
- 然后通过两个condition变量, 一个用于在buffer非空时等待, 一个用于buffer未满时等待
- 上面使用while循环将await包围, 这是为了防止在使用Signal&Condition时产生signal stealers问题.
- 以上方法可以安全地在多个线程中被调用
还有一个例子, 用于协调多个线程按固定顺序进行输出
public class TestSequentialThreads {
private final Lock lock = new ReentrantLock();
private final Condition[] conditions = {lock.newCondition(), lock.newCondition(), lock.newCondition()};
private int count = 0;
public void action(int i) {
while (true) {
print(i + " wait lock");
lock.lock();
print(i + " has lock");
try {
while (count != i) {
print(i + " await");
conditions[i].await();
}
print("===== " + i + " =====");
Thread.sleep(500);
count = (count + 1) % 3;
int j = (i + 1) % 3;
print(i + " signal " + j);
conditions[j].signal();
} catch (InterruptedException e) {
print(i + " InterruptedException");
} finally {
print(i + " unlock");
lock.unlock();
}
}
}
public static void main(String[] args) {
TestSequentialThreads ts = new TestSequentialThreads();
new Thread(()->ts.action(0)).start();
new Thread(()->ts.action(2)).start();
new Thread(()->ts.action(1)).start();
new Thread(()->ts.action(1)).start();
new Thread(()->ts.action(0)).start();
new Thread(()->ts.action(2)).start();
}
public static void print(String str) {
System.out.println(str);
}
}
如果是使用wait()和notify()的话, 就要写成这样, 这种情况下, 运行时notify()随机通知的线程, 是有可能不满足而跳过的.
public class DemoThreadWait2 {
private Object obj = 0;
private int pos = 1;
public void one(int i) {
synchronized (obj) {
if (pos == i) {
System.out.println("T-" + i);
pos = i % 3 + 1;
} else {
// System.out.println(".");
}
obj.notify();
try {
obj.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
DemoThreadWait2 demo = new DemoThreadWait2();
new Thread(()->{
while(true) {
demo.one(1);
}
}).start();
new Thread(()->{
while(true) {
demo.one(2);
}
}).start();
new Thread(()->{
while(true) {
demo.one(3);
}
}).start();
}
}
Java线程同步的Monitor机制(Lock配合Condition)的更多相关文章
- (删)Java线程同步实现二:Lock锁和Condition
在上篇文章(3.Java多线程总结系列:Java的线程同步实现)中,我们介绍了用synchronized关键字实现线程同步.但在Java中还有一种方式可以实现线程同步,那就是Lock锁. 一.同步锁 ...
- 转:C# 线程同步技术 Monitor 和Lock
原文地址:http://www.cnblogs.com/lxblog/archive/2013/03/07/2947182.html 今天我们总结一下 C#线程同步 中的 Monitor 类 和 Lo ...
- Java线程同步_1
Java线程同步_1 synchronized 该同步机制的的核心是同步监视器,任何对象都可以作为同步监视器,代码执行结束,或者程序调用了同步监视器的wait方法会导致释放同步监视器 synchron ...
- Java线程同步之一--AQS
Java线程同步之一--AQS 线程同步是指两个并发执行的线程在同一时间不同时执行某一部分的程序.同步问题在生活中也很常见,就比如在麦当劳点餐,假设只有一个服务员能够提供点餐服务.每个服务员在同一时刻 ...
- JAVA - 线程同步和线程调度的相关方法
JAVA - 线程同步和线程调度的相关方法 wait():使一个线程处于等待(阻塞)状态,并且释放所持有的对象的锁:wait是Object类的方法,对此对象调用wait方法导致本线程放弃对象锁,进入等 ...
- Java线程同步的四种方式详解(建议收藏)
Java线程同步属于Java多线程与并发编程的核心点,需要重点掌握,下面我就来详解Java线程同步的4种主要的实现方式@mikechen 目录 什么是线程同步 线程同步的几种方式 1.使用sync ...
- java 线程同步 原理 sleep和wait区别
java线程同步的原理java会为每个Object对象分配一个monitor, 当某个对象(实例)的同步方法(synchronized methods)被多个线程调用时,该对象的monitor将负责处 ...
- java线程 同步临界区:thinking in java4 21.3.5
java线程 同步临界区:thinking in java4 21.3.5 thinking in java 4免费下载:http://download.csdn.net/detail/liangru ...
- 多线程状态与优先级、线程同步与Monitor类、死锁
一.线程状态 二.线程优先级 三.初步尝试多线程 class Program { static void Main(string[] args) { while (true) { MessagePri ...
随机推荐
- Spring IOC 总结
IOC 简介 IOC是(Inversion of Control,控制反转)的简写.Spring提供IOC容器,将对象间的依赖关系交由Spring进行控制,避免硬编码所造成的的过度程序耦合.它由DI( ...
- Python3链接Oracle
1. 说明 本篇主要参见与cx_Oracle安装 全部操作均在root用户下完成 2. 下载Oracle Instant Client客户端 依据系统,在Oracle Instant Client下载 ...
- Linux-firewall防火墙
systemctl status firewalld firewall-cmd --zone=public --list-ports ##查看已开放的端口 2.添加5901端口到白名单 执行 fire ...
- JS正则表达式提取数字
/** * [参数str] * @type {var String} * return 30 */ var str = "ren民BI30kuai" console.log(str ...
- idea中新增package总是嵌套的解决方法
在idea中创建package,为了方便会将com.xx.xx作为一个package,下面添加对应的子package.比如service,config等.但是当我创建是总是会嵌套在下面变成了com.x ...
- Jenkins - 扯淡篇
目录 什么是持续集成 持续集成的概念 持续交付 持续部署 流程 当没有Jenkins的时候... 什么是Jenkins 返回Jenkins目录 什么是持续集成 由于懒得写,所以本段摘自阮一峰老师的博客 ...
- git免密
免账号密码输入 git clone https://lichuanfa%40gitcloud.com.cn:lcf13870752164@git.c.citic/Citic-Data/bigdata_ ...
- go 学习 (二):基本语法
一.数据类型 布尔型:布尔型的值只可以是常量 true 或者 false.eg:var bo bool = true.布尔型无法参与数值运算,也无法与其他类型进行转换 数字类型:整型 int .浮点型 ...
- PHP截取字符串函数substr()函数实例用法详解
在PHP中有一项非常重要的技术,就是截取指定字符串中指定长度的字符.PHP对于字符串截取可以使用PHP预定义函数substr()函数来实现.下面就来介绍一下substr()函数的语法及其应用. sub ...
- Redis 高可用架构设计(转载)
转载自:https://mp.weixin.qq.com/s?__biz=MzA3NDcyMTQyNQ==&mid=2649263292&idx=1&sn=b170390684 ...