同步和协作工具类

一、读写锁ReentrantReadWriteLock

ReadWriteLock接口的定义为:

public interface ReadWriteLock {
Lock readLock();
Lock writeLock();
}

读操作使用读锁,写操作使用写锁。只有"读-读"操作是可以并行的,"读-写"和"写-写"都不行。

始终只有一个线程能进行写操作,在获取写锁时,只有没有任何线程持有任何锁才可以获取到,

在持有写锁时,其他任何线程都获取不到任何锁。在没有其他线程持有写锁的情况下,多个线程可以获取和持有读锁。

ReentrantReadWriteLock的两个构造方法:

public ReentrantReadWriteLock() {
this(false);
}
public ReentrantReadWriteLock(boolean fair) {
sync = fair ? new FairSync() : new NonfairSync();
readerLock = new ReadLock(this);
writerLock = new WriteLock(this);
}

其中fair表示是否公平。

二、信号量Semaphore

信号量类Semaphore类用来限制对资源并发访问的线程数,构造方法:

//permits表示许可数量
public Semaphore(int permits)
//fair表示是否公平
public Semaphore(int permits, boolean fair)

Semaphore方法与锁类似,主要有两类方法,获取许可和释放许可:

//阻塞获取许可
public void acquire() throws InterruptedException
//阻塞获取许可,不响应中断
public void acquireUninterruptibly()
//批量获取多个许可
public void acquire(int permits) throws InterruptedException
public void acquireUninterruptibly(int permits)
//尝试获取
public boolean tryAcquire()
//限定等待时间获取
public boolean tryAcquire(int permits, long timeout,
TimeUnit unit) throws InterruptedException
//释放许可
public void release()

限制并发访问数量不超过100的例子:

public class AccessControlService {
public static class ConcurrentLimitException extends RuntimeException {
}
private static final int MAX_PERMITS = 100;
private Semaphore permits = new Semaphore(MAX_PERMITS, true);
public boolean login(String name, String password) {
//每次acquire都会消耗一个许可
if (!permits.tryAcquire()) {
throw new ConcurrentLimitException();
}
return true;
}
public void logout(String name) {
permits.release();
}
}
Semaphore permits = new Semaphore(1);
permits.acquire();
//程序会阻塞在第二个acquire调用
permits.acquire();
System.out.println("acquired");

信号量也是基于AQS实现的。

三、倒计时门栓CountDownLatch

用于需要线程同步的情景。该类相当于一个门栓,一开始是关闭的,所有希望通过该门的线程都需要等待,

然后开始倒计时,倒计时变为0的时候,门栓打开,所有线程通过,它是一次性的,打开后不能关闭。构造函数:

public CountDownLatch(int count)

与多个线程的协作方法:

//检查计数是否为0如果大于0就等待。await可以被中断,也可以设置最长等待时间
public void await() throws InterruptedException
public boolean await(long timeout, TimeUnit unit) throws InterruptedException
//countDown检查计数,如果已经为0,直接返回,否则减少计数,如果新的计数变为0,
//则唤醒所有线程
public void countDown()
public class RacerWithCountDownLatch {
static class Racer extends Thread {
CountDownLatch latch;
public Racer(CountDownLatch latch) {
this.latch = latch;
}
@Override
public void run() {
try{
this.latch.await();
System.out.println(getName() + " start run "
+ System.currentTimeMillis());
} catch (Exception e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) throws InterruptedException {
int num = 10;
CountDownLatch latch = new CountDownLatch(1);
Thread[] threads = new Thread[num];
for (int i = 0; i < num; i++) {
threads[i] = new Racer(latch);
threads[i].start();
}
Thread.sleep(1000);
latch.countDown();
/*Thread-0 start run 1545714108398
Thread-3 start run 1545714108398
Thread-4 start run 1545714108398
Thread-5 start run 1545714108398
Thread-6 start run 1545714108398
Thread-7 start run 1545714108399
Thread-8 start run 1545714108399
Thread-9 start run 1545714108399*/
}
}

四、循环栅栏CyclicBarrier

所有线程在到达栅栏后都需要等待其他线程,等所有线程都到达后再一起通过,

它是循环的,可以用作重复的同步。构造方法:

//parties参与线程个数
public CyclicBarrier(int parties)
//barrierAction表示所有线程到达栅栏后,所有线程执行下一步动作前,
//运行参数中的动作,这个动作由最后一个到达栅栏的线程执行
public CyclicBarrier(int parties, Runnable barrierAction)

主要方法:

//等待其他线程到达栅栏,调用await后表示自己已经到达,如果是最后一个到达的,就执行可选命令,执行完毕后,唤醒所有等待的线程,然后重置内部的同步计数
public int await() throws InterruptedException, BrokenBarrierException
public int await(long timeout, TimeUnit unit) throws InterruptedException,
BrokenBarrierException, TimeoutException

注意:在CyclicBarrier中,参与的线程是互相影响的,只要有其中的一个线程在调用await时被中断或者超时了,

栅栏就会被破坏。此外,如果栅栏动作抛出了异常,栅栏也会被破坏。被破坏后,所有在调用的await线程就会退出,

抛出BrokenBarrierException。

五、ThreadLocal

1.基本概念和用法

线程本地变量:每个线程都有同一个变量的独特拷贝。ThreadLocal是一个泛型类,

接受一个类型参数T,它只有一个空的构造方法,有两个主要的public方法:

//获取值
public T get()
//设置值
public void set(T value)
public class ThreadLocalBasic {
static ThreadLocal<Integer> local = new ThreadLocal<Integer>();
public static void main(String[] args) throws InterruptedException {
Thread child = new Thread(){
@Override
public void run() {
System.out.println("child thread initial " + local.get()); //null
local.set(200);
System.out.println("child thread final: " + local.get()); //
}
};
local.set(100);
child.start();
child.join();
System.out.println("Main thread final : " + local.get()); //
}
}

从上面的例子可以看出,一个线程本地变量,在每个线程都有自己的独立值。

ThreadLocal的其他方法:

//用于提供初始值
protected T initialValue()
//删除当前线程的对应值,删掉后再次调用get就会获取初始值
public void remove()

应用:是实现线程安全、减少竞争的一种方案。

Java笔记(十八)同步和协作工具类的更多相关文章

  1. java第十九天,Collections集合工具类的使用

    Collections Java中集合都实现了Collection接口,那么针对集合一些特定的功能,有没有一个接口或类能够统一的集成一些集合必要的功能呢?当然能.它就是--Collections集合工 ...

  2. Java笔记(十八)……包

    概述 对类文件进行分类管理. 给类提供多层命名空间. 写在程序文件的第一行. 类名的全称的是 包名.类名. 包也是一种封装形式. 访问权限 引用<The Complete Reference&g ...

  3. python3.4学习笔记(十八) pycharm 安装使用、注册码、显示行号和字体大小等常用设置

    python3.4学习笔记(十八) pycharm 安装使用.注册码.显示行号和字体大小等常用设置Download JetBrains Python IDE :: PyCharmhttp://www. ...

  4. 《java面试十八式》--引子

    爪哇城中   “喂,你等等我啊”少女气喘吁吁的喊道   “大小姐,你可快点吧,报名马上就要结束了.”   这是爪哇城一年一度的大选比赛,被选上的人会留下来任职,享有名誉和金钱,所以大家都在积极准备. ...

  5. java调用kettle的job和transfer工具类

    package com.woaiyitiaocai.util; import java.util.Map; import java.util.UUID; import org.apache.log4j ...

  6. Java Class与反射相关的一些工具类

    package com.opslab.util; import org.apache.log4j.Logger; import java.io.File;import java.io.IOExcept ...

  7. Java语言Lang包下常用的工具类介绍_java - JAVA

    文章来源:嗨学网 敏而好学论坛www.piaodoo.com 欢迎大家相互学习 无论你在开发哪中 Java 应用程序,都免不了要写很多工具类/工具函数.你可知道,有很多现成的工具类可用,并且代码质量都 ...

  8. Java基础学习笔记十八 异常处理

    什么是异常?Java代码在运行时期发生的问题就是异常. 在Java中,把异常信息封装成了一个类.当出现了问题时,就会创建异常类对象并抛出异常相关的信息(如异常出现的位置.原因等). 异常的继承体系 在 ...

  9. Java并发(十六):并发工具类——Exchanger

    Exchanger(交换者)是一个用于线程间协作的工具类.Exchanger用于进行线程间的数据交换.它提供一个同步点,在这个同步点两个线程可以交换彼此的数据.这两个线程通过exchange方法交换数 ...

随机推荐

  1. bzoj 4816

    这题是莫比乌斯反演的典型题也是很有趣的题. 题意:求,其中f为为斐波那契数列 那么首先观察一下指数,发现是我们熟悉的形式,可以转化成这样的形式: 令T=kd,且假设n<m,有: 令 则原式= 这 ...

  2. 检查URL的可用性脚本

    #!/bin/bash check_url() { HTTP_CODE=$(curl -o /dev/ -s -) ];then echo "Warning: $1 Access failu ...

  3. 80端口被占用 导致apach无法启动问题

    1.查找是哪个程序占用了80端口 netstat -ano 列出所有进程 观察 “本地地址” 列 找到对应的PID 我这里是4 简单的办法,打开任务管理器,查看PID是4的 是哪个进程. 发现是Sys ...

  4. C++11 中的function和bind、lambda用法

    std::function 1. std::bind绑定一个成员函数 #include <iostream> #include <functional> struct Foo ...

  5. 解决Windows Server 2008R2通过计划任务定时执行bat文件,显示成功但实际未执行

    前段时间在Windows Server 2008安装了一套基于MySQL数据库的软件,处于数据安全的考虑,希望每天能够自动进行数据库备份.我在别人脚本的基础上自己写了一个数据库备份的bat脚本,双击该 ...

  6. flanneld,flannel和cni逐步深入

    这个问题,现在慢慢搞定. 把这两者的关系搞清楚了. 还有kubeadm join和手工安装Node的故事, 也没那么玄乎~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ htt ...

  7. QQ登录用到的URL

    //QQ 登陆页面的URL,client_id就是APP ID,会返回一个codehttps://graph.qq.com/oauth2.0/authorize?response_type=code& ...

  8. [转] 前后端分手大师——MVVM 模式

    之前对 MVVM 模式一直只是模模糊糊的认识,正所谓没有实践就没有发言权,通过这两年对 Vue 框架的深入学习和项目实践,终于可以装B了有了拨开云雾见月明的感觉. Model–View–ViewMod ...

  9. [转]centos安装autossh

    centos安装autossh $ sudo yum install wget gcc make$ wget http://www.harding.motd.ca/autossh/autossh-1. ...

  10. 如何使用maven进行avro序列化

    maven导入avro: <dependency> <groupId>org.apache.avro</groupId> <artifactId>avr ...