构建自定义同步工具

一、通过轮询与休眠的方式实现简单的有界缓存

public void put(V v) throws InterruptedException {
while (true) { //轮询
synchronized (this) { //加锁等待
if (!isFull()) { //如果缓存没满 则执行插入
doPut(v);
return;
}
}
Thread.sleep(SLEEP_GRANULARITY); //如果缓存满了 线程等待一段时间后继续轮询
}
} public V take() throws InterruptedException {//同理 相反
while (true) {
synchronized (this) {
if (!isEmpty())
return doTake();
}
Thread.sleep(SLEEP_GRANULARITY);
}
}

二、通过条件队列:wait notify方法

  注意:实际逻辑与上面没区别;且无法通过轮询和休眠方式实现的,也无法通过条件队列实现

  

    // BLOCKS-UNTIL: not-full
public synchronized void put(V v) throws InterruptedException {
while (isFull()) {
wait(); //等待:释放锁且等待;等待通知后获取锁 并继续判断条件谓词
}
doPut(v);
notifyAll(); //通知:通知释放所有等待
}
 
// 问题,不好分析,在put 和 take操作后,通知了所有等待条件队列
// 而且让所有 等待都要去判断 条件谓词;
// 如我put完成,只需要通知take正在等待的,而无需通知put正在等待的(反之亦然),因为只有take后,缓存才有可能不是满的,才要去判断条件谓词
public synchronized V take() throws InterruptedException {
while (isEmpty()) {
wait();
}
V v = doTake();
notifyAll();
return v;
}

三、内置条件队列的使用

   条件谓词:对象在哪个条件下等待

  条件队列:每次唤醒时,必须重新检查条件谓词

四、显式锁Condition:自定义条件队列

  内置条件队列:每个内置锁只能有一个相关联的条件队列

    所以:多个线程在一个条件队列上等待不同的条件谓词不好解决

    方法:wait、notify、notifyAll

  而Condition:每个Lock,可以有任意的Condition

    方法:await、signal、signalAll

优化内置的条件队列:

public class ConditionBoundedBuffer <T> {
protected final Lock lock = new ReentrantLock();
// 条件谓词 未满
private final Condition notFull = lock.newCondition();
// 条件谓词 非空
private final Condition notEmpty = lock.newCondition();
private static final int BUFFER_SIZE = 100;
private final T[] items = (T[]) new Object[BUFFER_SIZE];
private int tail, head, count; // BLOCKS-UNTIL: notFull
public void put(T x) throws InterruptedException {
lock.lock();
try {
while (count == items.length)
notFull.await(); //当缓存满了,等待有线程取出缓存;等待未满谓词通知(只有take才有可能让缓存不满)
items[tail] = x;
if (++tail == items.length)
tail = 0;
++count;
notEmpty.signal(); //当插入成功,通知释放非空谓词
} finally {
lock.unlock();
}
} // BLOCKS-UNTIL: notEmpty
public T take() throws InterruptedException {
lock.lock();
try {
while (count == 0)
notEmpty.await(); //当缓存为空,等待插入;等待非空谓词通知
T x = items[head];
items[head] = null;
if (++head == items.length)
head = 0;
--count;
notFull.signal(); //当取出一个,通知释放未满谓词
return x;
} finally {
lock.unlock();
}
}
}

五、AQS:AbstractQueuedSynchronizer

  先来看一个例子:

//通过 lock+条件队列实现信号量Semaphore 许可
public class SemaphoreOnLock {
private final Lock lock = new ReentrantLock();
// CONDITION PREDICATE: permitsAvailable (permits > 0)
private final Condition permitsAvailable = lock.newCondition();
private int permits; SemaphoreOnLock(int initialPermits) {
lock.lock();
try {
permits = initialPermits;
} finally {
lock.unlock();
}
} // BLOCKS-UNTIL: permitsAvailable
public void acquire() throws InterruptedException {
lock.lock();
try {
while (permits <= 0) //当许可<0时 等待 许可释放
permitsAvailable.await();
--permits;
} finally {
lock.unlock();
}
} public void release() {
lock.lock();
try {
++permits; //释放许可
permitsAvailable.signal();
} finally {
lock.unlock();
}
}
}

  AQS负责管理同步器类中的状态,它管理了一个整数状态信息;

  如:通过AQS实现简单的闭锁

public class OneShotLatch {
private final Sync sync = new Sync(); public void signal() {
sync.releaseShared(0);
} public void await() throws InterruptedException {
sync.acquireSharedInterruptibly(0);
} private class Sync extends AbstractQueuedSynchronizer { //继承AQS
protected int tryAcquireShared(int ignored) {
// 判断状态
return (getState() == 1) ? 1 : -1;
} protected boolean tryReleaseShared(int ignored) {
//释放状态
setState(1); // 打开闭锁
return true; // 其他线程可以获取该闭锁 }
}
}

小结

  1.最好使用现有的类库来构建状态类:如缓存

  2.如果现有类不能满足功能:如从一个空队列中删除或获取元素会抛出异常,当然也有阻塞等待

    那么使用Lock、内置的条件队列、Condition显式的条件队列、AQS等来构建自己的同步器也是可行的

    内置条件队列与内置锁关联、显式的条件队列与Lock关联

java并发编程(7)构建自定义同步工具及条件队列的更多相关文章

  1. Java并发编程的4个同步辅助类

    Java并发编程的4个同步辅助类(CountDownLatch.CyclicBarrier.Semphore.Phaser) @https://www.cnblogs.com/lizhangyong/ ...

  2. Java并发编程:线程的同步

    Java并发编程:线程的同步 */--> code {color: #FF0000} pre.src {background-color: #002b36; color: #839496;} J ...

  3. Java并发编程的4个同步辅助类(CountDownLatch、CyclicBarrier、Semaphore、Phaser)

    我在<JDK1.5引入的concurrent包>中,曾经介绍过CountDownLatch.CyclicBarrier两个类,还给出了CountDownLatch的演示案例.这里再系统总结 ...

  4. Java并发编程的4个同步辅助类(CountDownLatch、CyclicBarrier、Semphore、Phaser)

    我在<jdk1.5引入的concurrent包>中,曾经介绍过CountDownLatch.CyclicBarrier两个类,还给出了CountDownLatch的演示案例.这里再系统总结 ...

  5. 【Java并发编程实战】-----“J.U.C”:CLH队列锁

    在前面介绍的几篇博客中总是提到CLH队列,在AQS中CLH队列是维护一组线程的严格按照FIFO的队列.他能够确保无饥饿,严格的先来先服务的公平性.下图是CLH队列节点的示意图: 在CLH队列的节点QN ...

  6. 【Java并发编程实战】-----“J.U.C”:CLH队列锁

    在前面介绍的几篇博客中总是提到CLH队列,在AQS中CLH队列是维护一组线程的严格依照FIFO的队列.他可以确保无饥饿,严格的先来先服务的公平性.下图是CLH队列节点的示意图: 在CLH队列的节点QN ...

  7. 《java并发编程实战》读书笔记11--构建自定义的同步工具,条件队列,Condition,AQS

    第14章 构建自定义的同步工具 本章将介绍实现状态依赖性的各种选择,以及在使用平台提供的状态依赖机制时需要遵守的各项规则. 14.1 状态依赖性的管理 对于并发对象上依赖状态的方法,虽然有时候在前提条 ...

  8. Java并发系列[4]----AbstractQueuedSynchronizer源码分析之条件队列

    通过前面三篇的分析,我们深入了解了AbstractQueuedSynchronizer的内部结构和一些设计理念,知道了AbstractQueuedSynchronizer内部维护了一个同步状态和两个排 ...

  9. Java并发编程(八)同步容器

    为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器.并发容器.阻塞队列.Synchronizer(比如CountDownLatch) 一.为什么会出现同步容器? ...

随机推荐

  1. css细节复习笔记——基本视觉格式化

    css包含如此开放.如此强大的一个模型,对于这样一个模型,可以有无数种方法结合应用各种属性,可以得到的效果数不胜数. 基本框 css假定每个元素都会生成一个或多个矩形框,这称为元素框.各元素中心有一个 ...

  2. Spring 中aop切面注解实现

    spring中aop的注解实现方式简单实例   上篇中我们讲到spring的xml实现,这里我们讲讲使用注解如何实现aop呢.前面已经讲过aop的简单理解了,这里就不在赘述了. 注解方式实现aop我们 ...

  3. pkuwc 前的任务计划

    菜鸡 wxw 的计划(肯定会咕咕咕 12.27 luogu P4244 [SHOI2008]仙人掌图 II(咕咕咕 luogu P4246 [SHOI2008]堵塞的交通 (没有咕! luogu P1 ...

  4. mxonline实战15,用户中心:我的课程, 我的收藏,我的消息,登出和点击数以及收藏数

     对应github地址:第15天     一.  我的课程   1. 继承usercenter-base页面 2. 编写url, view

  5. nginx的几种负载均衡策略

    转自https://www.cnblogs.com/1214804270hacker/p/9325150.html 一.关于Nginx的负载均衡 在服务器集群中,Nginx起到一个代理服务器的角色(即 ...

  6. eclipse远程debug服务器上的项目(Tomcat),打开、关闭及常见错误汇总

    我们工作中,有时候测试/生产环境,出现的结果会与我们预计的不一样,只看代码又看不出去问题所在,这个时候就需要远程debug下服务器上的项目. 注意:(1)需要debug的代码,本机代码需与服务器上一致 ...

  7. java中的jdk配置详解:

    1.配值系统变量"JAVA_HOME" 变量名JAVA_HOME: 指向:JDK(java开发工具包)的安装路径 目的:使用JDK安装目录时,可以直接通过”%JAVA_HOME%“ ...

  8. iOS开发~制作同时支持armv7,armv7s,arm64,i386,x86_64的静态库.a以及 FrameWork 的创建

    armv7,armv7s,arm64,i386,x86_64 详解 一.概要 平时项目开发中,可能使用第三方提供的静态库.a,如果.a提供方技术不成熟,使用的时候就会出现问题,例如: 在真机上编译报错 ...

  9. Java Web 热部署

    热部署有多种方案,下面的方案是其中的一种. 暂时还没找到一种令人满意的方案. 1,配置WEB Server 去这里 (https://tomcat.apache.org/download-90.cgi ...

  10. pycharm+gitee

    Git操作 前言: 由于各种原因,很多时候我们写代码的电脑并不会随身携带,所以有的时候突发灵感想继续写代码就变得难以实现.相信大部分同学对此都有了解,那就通过代码托管平台来管理.原本想用GitHub来 ...