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 ...
随机推荐
- 腾讯微服务框架Tars的初体验
最近研究了一下腾讯的微服务体系开发框架. 官方的搭建过程:https://github.com/TarsCloud/Tars/blob/master/Install.zh.md 自己填的坑: 不得不说 ...
- springboot+mybatis +yml文件配置多数据源
记录一下java开发中多数据源的配置过程, 参考博客:https://blog.csdn.net/weinichendian/article/details/72903757,我在这里进行了整理,使用 ...
- 27.centos7基础学习与积累-013-文件和目录的权限
从头开始积累centos7系统运用 大牛博客: https://blog.51cto.com/yangrong/p5 https://blog.oldboyedu.com/ 文件的权限 rw-r--r ...
- python3中用HTMLTestRunner.py报ImportError: No module named 'StringIO'的解决方法:
全文转载至:http://www.cnblogs.com/testyao/p/5658200.html python3中用HTMLTestRunner.py报ImportError: No modul ...
- windows(hexo)使用git时出现:warning: LF will be replaced by CRLF
hexo出现warning: LF will be replaced by CRLF git config --global core.autocrlf false //禁用自动转换
- 【pathon基础】初识python
一.python的起源 作者:Guido van Rossum(龟叔) 设计原则:优雅,简单,明确 二.解释型语言VS编译型语言 1.解释型语言:C#.python step1:程序员写代码: ste ...
- php过滤敏感词
<?php /** * 敏感词过滤工具类 * 使用方法 * echo FilterTools::filterContent("你妈的我操一色狼杂种二山食物"," ...
- STLNormalFunc
#include <iostream> #include <vector> using namespace std; void main_1() { vector<int ...
- Centos7 修改为阿里的yum源
1.备份本地yum源 mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo_bak 2.获取阿里yum源配置文件 ...
- fitnesse的安装
最近项目组有个单独的功能模块需要写自动化,由于是测试接口,我本来是想用之前那个项目组使用的robot framework+python,但是呢,项目组领导觉得,目前项目开发语言是java,相应的自动化 ...