您好,我是湘王,这是我的博客园,欢迎您来,欢迎您再来~

之前说过,AQS(抽象队列同步器)是Java锁机制的底层实现。既然它这么优秀,是骡子是马,就拉出来溜溜吧。

首先用重入锁来实现简单的累加,就像这样:

/**
* 用重入锁实现累加
*
* @author 湘王
*/
public class MyLockTest {
private final Lock lock = new ReentrantLock();
private int value;
public int getNext() {
lock.lock();
try {
value++;
} finally {
lock.unlock();
}
return value;
}
public static void main(String[] args) {
MyLockTest myLock = new MyLockTest();
for (int i = 0; i < 5; i++) {
new Thread(new Runnable() {
@Override
public void run() {
for (int i = 0; i < 5; i++) {
System.out.println(myLock.getNext());
}
}
}).start();
}
}
}

运行结果显示数据有重复:

这么简单的计算都能出现重复,这肯定是无法接受的。

再用独占锁来试试看:

/**
* 利用AQS实现自定义独占锁
*
* @author 湘王
*/
public class MyExclusiveLock implements Lock {
@Override
public void lock() { } @Override
public void lockInterruptibly() throws InterruptedException { } @Override
public boolean tryLock() {
return false;
} @Override
public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
return false;
} @Override
public void unlock() { } @Override
public Condition newCondition() {
return null;
}
}

可以看到,实现lock接口,就需要实现若干自定义的接口。然后以内部类继承AQS的方式,实现排他锁,昨天也说过,AQS中tryAcquire()和tryRelease()是一一对应的,也就是也管获取,一个管释放,所以代码是:

/**
* 内部类继承AQS的方式,实现排他锁
*/
private static class SyncHelper extends AbstractQueuedSynchronizer {
private static final long serialVersionUID = -7666580981453962426L; /**
* 第一个线程进来,拿到锁就返回true;后面的线程进来,拿不到锁就返回false
*/
@Override
protected boolean tryAcquire(int arg) {
// 获取资源状态
int state = getState();
if (0 == state) {// 如果没有线程拿到资源的锁
if (compareAndSetState(0, arg)) {
// 保存当前持有同步锁的线程
setExclusiveOwnerThread(Thread.currentThread());
return true;
}
} else if (Thread.currentThread() == getExclusiveOwnerThread()) {
// 如果当前线程再次进来,state + 1,可重入
// 如果这里没有这个判断,那么程序会卡死
setState(state + arg);
return true;
}
return false;
} /**
* 锁的获取和释放需要一一对应
*/
@Override
protected boolean tryRelease(int arg) {
// 获取资源状态
int state = getState();
// 返回最后一个通过setExclusiveOwnerThread()方法设置过的线程,或者null
if (Thread.currentThread() != getExclusiveOwnerThread()) {
throw new RuntimeException();
}
setState(state - arg);
if (0 == state) {
setExclusiveOwnerThread(null);
return true;
}
return false;
} protected Condition newCondition() {
return new ConditionObject();
}
}

然后再用AQS实现lock接口的方法:

/**
* 利用AQS实现自定义独占锁
*
* @author 湘王
*/
public class MyExclusiveLock implements Lock {
private final SyncHelper synchepler = new SyncHelper(); @Override
public void lock() {
synchepler.acquire(1);
} @Override
public void lockInterruptibly() throws InterruptedException {
synchepler.acquireInterruptibly(1);
} @Override
public boolean tryLock() {
return synchepler.tryAcquire(1);
} @Override
public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
return synchepler.tryAcquireNanos(1, unit.toNanos(time));
} @Override
public void unlock() {
synchepler.release(1);
} @Override
public Condition newCondition() {
return synchepler.newCondition();
} /**
* 内部类继承AQS的方式,实现排他锁
*/
private static class SyncHelper extends AbstractQueuedSynchronizer {
private static final long serialVersionUID = -7666580981453962426L; /**
* 第一个线程进来,拿到锁就返回true;后面的线程进来,拿不到锁就返回false
*/
@Override
protected boolean tryAcquire(int arg) {
// 获取资源状态
int state = getState();
if (0 == state) {// 如果没有线程拿到资源的锁
if (compareAndSetState(0, arg)) {
// 保存当前持有同步锁的线程
setExclusiveOwnerThread(Thread.currentThread());
return true;
}
} else if (Thread.currentThread() == getExclusiveOwnerThread()) {
// 如果当前线程再次进来,state + 1,可重入
// 如果这里没有这个判断,那么程序会卡死
setState(state + arg);
return true;
}
return false;
} /**
* 锁的获取和释放需要一一对应
*/
@Override
protected boolean tryRelease(int arg) {
// 获取资源状态
int state = getState();
// 返回最后一个通过setExclusiveOwnerThread()方法设置过的线程,或者null
if (Thread.currentThread() != getExclusiveOwnerThread()) {
throw new RuntimeException();
}
setState(state - arg);
if (0 == state) {
setExclusiveOwnerThread(null);
return true;
}
return false;
} protected Condition newCondition() {
return new ConditionObject();
}
}
}

然后再运行测试:

/**
* 实现Lock接口方法并运行排他锁测试
*
* @author 湘王
*/
public class MyExclusiveLockTester {
// 用自定义AQS独占锁实现
private Lock lock = new MyExclusiveLock();
private int value; public int accmulator() {
lock.lock();
try {
++value;
} finally {
lock.unlock();
} return value;
} public static void main(String[] args) throws InterruptedException {
MyExclusiveLockTester test = new MyExclusiveLockTester();
for (int i = 0; i < 5; i++) {
new Thread(new Runnable() {
@Override
public void run() {
for (int i = 0; i < 5; i++) {
System.out.println(test.accmulator());
}
}
}).start();
}
}
}

可以看到,结果无论怎么样都不会再重复了。

这个只是简单的累加,接下来用AQS来实现一个实际的生活场景。比如周末带女票或男票去步行街吃饭,这时候人特别多,需要摇号,而且一次只能进去三张号(不按人头算,按叫到的号来算),该怎么实现呢?

可以顺着这个思路:摇号机虽有很多号,但它本质上是个共享资源,很多人可以共享,但是每次共享的数量有限。这其实就是个可以指定数量的共享锁而已。

既然有了思路,那接下来就好办了。

/**
* 利用AQS实现自定义共享锁
*
* @author 湘王
*/
public class MyShareLock implements Lock {
@Override
public void lock() {
} @Override
public void lockInterruptibly() throws InterruptedException {
} @Override
public boolean tryLock() {
return false;
} @Override
public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
return false;
} @Override
public void unlock() {
} @Override
public Condition newCondition() {
return null;
}
}

还是一样实现Lock接口,但这次是用AQS实现共享锁。

/**
* 内部类继承AQS实现共享锁
*
*/
private static class SyncHelper extends AbstractQueuedSynchronizer {
private static final long serialVersionUID = -7357716912664213942L; /**
* count表示允许几个线程能同时获得锁
*/
public SyncHelper(int count) {
if (count <= 0) {
throw new IllegalArgumentException("锁资源数量必须大于0");
}
// 设置资源总数
setState(count);
} /**
* 一次允许多少个线程进来,允许数量的线程都能拿到锁,其他的线程进入队列
*/
@Override
protected int tryAcquireShared(int acquires) {
// 自旋
for (;;) {
int state = getState();
int remain = state - acquires;
// 判断剩余锁资源是否已小于0或者CAS执行是否成功
if (remain < 0 || compareAndSetState(state, remain)) {
return remain;
}
}
} /**
* 锁资源的获取和释放要一一对应
*/
@Override
protected boolean tryReleaseShared(int releases) {
// 自旋
for (;;) {
// 获取当前state
int current = getState();
// 释放状态state增加releases
int next = current + releases;
if (next < current) {// 溢出
throw new Error("Maximum permit count exceeded");
}
// 通过CAS更新state的值
// 这里不能用setState()
if (compareAndSetState(current, next)) {
return true;
}
}
} protected Condition newCondition() {
return new ConditionObject();
}
}

然后再来改造之前实现的接口:

/**
* 利用AQS实现自定义共享锁
*
* @author 湘王
*/
public class MyShareLock implements Lock {
public static int count;
private final SyncHelper synchepler = new SyncHelper(count); @Override
public void lock() {
synchepler.acquireShared(1);
} @Override
public void lockInterruptibly() throws InterruptedException {
synchepler.acquireSharedInterruptibly(1);
} @Override
public boolean tryLock() {
return synchepler.tryAcquireShared(1) > 0;
} @Override
public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
return synchepler.tryAcquireSharedNanos(1, unit.toNanos(time));
} @Override
public void unlock() {
synchepler.releaseShared(1);
} @Override
public Condition newCondition() {
return synchepler.newCondition();
} /**
* 内部类继承AQS实现共享锁
*
*/
private static class SyncHelper extends AbstractQueuedSynchronizer {
private static final long serialVersionUID = -7357716912664213942L; /**
* count表示允许几个线程能同时获得锁
*/
public SyncHelper(int count) {
if (count <= 0) {
throw new IllegalArgumentException("锁资源数量必须大于0");
}
// 设置资源总数
setState(count);
} /**
* 一次允许多少个线程进来,允许数量的线程都能拿到锁,其他的线程进入队列
*/
@Override
protected int tryAcquireShared(int acquires) {
// 自旋
for (;;) {
int state = getState();
int remain = state - acquires;
// 判断剩余锁资源是否已小于0或者CAS执行是否成功
if (remain < 0 || compareAndSetState(state, remain)) {
return remain;
}
}
} /**
* 锁资源的获取和释放要一一对应
*/
@Override
protected boolean tryReleaseShared(int releases) {
// 自旋
for (;;) {
// 获取当前state
int current = getState();
// 释放状态state增加releases
int next = current + releases;
if (next < current) {// 溢出
throw new Error("Maximum permit count exceeded");
}
// 通过CAS更新state的值
// 这里不能用setState()
if (compareAndSetState(current, next)) {
return true;
}
}
} protected Condition newCondition() {
return new ConditionObject();
}
}
}

接下来就该测试咱们需要的效果是否能实现了:

public class MyShareLockTester {
public static void main(String[] args) throws InterruptedException {
// 用自定义AQS共享锁实现
// 一次允许发放三把锁
MyShareLock.count = 3;
final Lock lock = new MyShareLock(); // 模拟20个客户端访问
for (int i = 0; i < 20; i++) {
new Thread(new Runnable() {
@Override
public void run() {
try {
lock.lock();
System.out.println("持有 " + Thread.currentThread().getName() + " 的客人可以进餐厅就餐");
// 每两次叫号之间间隔一段时间,模拟真实场景
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
// 使用完成释放锁
lock.unlock();
}
}
}).start();
}
}
}

这里有20个号,每次只能发放3张,运行之后就可以看到确实如此。

AQS是个很神奇也很好玩的东西,就像它的作者(也是除了高司令就是对Java影响最大的那个人,整个Java的多线程juc包代码就是他编写的)Doug Lea在AbstractQueuedSynchronizer的注释中所说:AQS只是一个框架,至于怎么玩,就是你的事了!


感谢您的大驾光临!咨询技术、产品、运营和管理相关问题,请关注后留言。欢迎骚扰,不胜荣幸~

Java多线程(6):锁与AQS(下)的更多相关文章

  1. Java 多线程:锁(三)

    Java 多线程:锁(三) 作者:Grey 原文地址: 博客园:Java 多线程:锁(三) CSDN:Java 多线程:锁(三) StampedLock StampedLock其实是对读写锁的一种改进 ...

  2. JAVA多线程与锁机制

    JAVA多线程与锁机制 1 关于Synchronized和lock synchronized是Java的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码 ...

  3. Java 多线程:锁(一)

    Java 多线程:锁(一) 作者:Grey 原文地址: 博客园:Java 多线程:锁(一) CSDN:Java 多线程:锁(一) CAS 比较与交换的意思 举个例子,内存有个值是 3,如果用 Java ...

  4. Java 多线程:锁(二)

    Java 多线程:锁(二) 作者:Grey 原文地址: 博客园:Java 多线程:锁(二) CSDN:Java 多线程:锁(二) AtomicLong VS LongAddr VS Synchroni ...

  5. java多线程之锁 -- 偏向锁、轻量级锁、自旋锁、重量级锁

    转载至:https://blog.csdn.net/zqz_zqz/article/details/70233767 之前做过一个测试,详情见这篇文章<多线程 +1操作的几种实现方式,及效率对比 ...

  6. java多线程----悲观锁与乐观锁

    java多线程中悲观锁与乐观锁思想 一.悲观锁 总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会阻塞直到它拿到锁(共享资源每次只给一个线 ...

  7. (转)java 多线程 对象锁&类锁

    转自:http://blog.csdn.net/u013142781/article/details/51697672 最近工作有用到一些多线程的东西,之前吧,有用到synchronized同步块,不 ...

  8. Java多线程--公平锁与非公平锁

    上一篇文章介绍了AQS的基本原理,它其实就是一个并发包的基础组件,用来实现各种锁,各种同步组件的.它包含了state变量.加锁线程.等待队列等并发中的核心组件,现在我们来看一下多线程获取锁的顺序问题. ...

  9. Java多线程系列——锁的那些事

    引入 Java提供了种类丰富的锁,每种锁因其特性的不同,在适当的场景下能够展现出非常高的效率. 下面先带大家来总体预览一下锁的分类图 java锁的具体实现类 1.乐观锁 VS 悲观锁 乐观锁与悲观锁是 ...

  10. java多线程编程——锁优化

    并发环境下进行编程时,需要使用锁机制来同步多线程间的操作,保证共享资源的互斥访问.加锁会带来性能上的损坏,似乎是众所周知的事情.然而,加锁本身不会带来多少的性能消耗,性能主要是在线程的获取锁的过程.如 ...

随机推荐

  1. 通过route , tracert , traceroute 查看本地路由配置及访问ip或域名时经过的路由信息

    转载请注明出处: 1.路由器和交换机的区别和过程 在windows 系统或linux 系统访问 外网ip 或域名时,都会通过层层的路由器,然后将请求转发到最终的目标服务器:因为互联网通过路由器实现公网 ...

  2. 如何定义 Java 的回调函数,与 JavaScript 回调函数的区别

    JavaScript 中的回调函数 在 JavaScript 中经常使用回调函数,比如:get 请求.post 请求等异步任务.在我们请求之前以及请求之后,都需要完成一些固定的操作,比如:请求之前先从 ...

  3. Hive存储格式之ORC File详解,什么是ORC File

    目录 概述 文件存储结构 Stripe Index Data Row Data Stripe Footer 两个补充名词 Row Group Stream File Footer 条纹信息 列统计 元 ...

  4. 【漏洞分析】KaoyaSwap 安全事件分析

    相关信息 KaoyaSwap 是 BSC 链上的一个自动做市商 AMM.然后,现在他们的官网 https://www.kaoyaswap.com/ 已经打不开了(如果我打开方式没错的话).所以就直接进 ...

  5. 安装Windows_server_2012_r2虚拟机步骤

    创建虚拟机 使用Windows_server_2012_r2镜像 网上搜索Windows产品密钥:TVNTG-VFJQ3-FQXFP-DVCP6-D3VJ8 点击完成,等待加载 选择第二个 等待安装 ...

  6. Altium Designer 18学习

    目录 目录 快捷键 通孔 敷铜 修改铜皮与导线之间的间隔 去除指定敷铜区域 DRC设计规则检查问题: 快捷键 EJC 快速跳转到器件 M 移动 CTRL+M 测量距离 通孔 敷铜 放置多边形平面 -- ...

  7. KingbaseES lag 和 lead 函数

    1.简介 lag与lead函数是跟偏移量相关的两个分析函数,通过这两个函数可以在一次查询中取出同一字段的前N行的数据(lag)和后N行的数据(lead)作为独立的列,从而更方便地进行进行数据过滤. 2 ...

  8. Docker容器网络基础总结

    ifconfig 之 docker0 基于Linux的虚拟网桥(通用网络设备的抽象) 虚拟网桥特点: 1. 可以设置IP地址 2.相当于拥有一个隐藏的虚拟网卡 docker0 的地址划分 IP: 17 ...

  9. 走进Redis-扯扯集群

    集群 为什么需要切片集群 已经有了管理主从集群的哨兵,为什么还需要推出切片集群呢?我认为有两个比较重要的原因: 当 Redis 上的数据一直累积的话,Redis 占用的内存会越来越大,如果开启了持久化 ...

  10. 使用SSH连接Windows Server

    之前发过一篇在Windows Server上启用SSH服务器的文章.最近正好有这个需求,需要使用密钥免密登录服务器,试了一下,发现之前的方法不行了.需要再修正一些文件权限. 需要使用Repair-Au ...