利用Atomic, ThreadLocal, 模仿AQS, ReentrantLock
/**
* @description 队列同步器,利用原子整形模仿AQS,非公平锁(简单自适应自旋)
* @since 2020/2/4
*/
public class QueueSynchronizer { private AtomicInteger state=new AtomicInteger(0);//0为可用,1为被用,2为已经重入一次,依此
private Thread onwThread;
private ConcurrentLinkedQueue<Thread> concurrentLinkedQueue=new ConcurrentLinkedQueue();//阻塞队列
//AQS队列中的头的thread永远为null,这里与AQS不一致,ConcurrentLinkedQueue不允许空值 //线程私有变量
private ThreadLocal<Integer> spinCount=new ThreadLocal<Integer>();//线程竞争失败后的自旋次数
private ThreadLocal<Integer> spinIncrement=new ThreadLocal<Integer>();//自旋成功后的自旋增量
private ThreadLocal<Integer> spinDecrement=new ThreadLocal<Integer>();//自旋成功后的自旋减量
private int sc;
private int si;
private int sd; //指定默认的自旋次数,成功增量,失败减量
public QueueSynchronizer(){
sc=4000;
si=900;
sd=700;
} //指定初始的自旋次数,成功增量,失败减量
public QueueSynchronizer(int spinCount, int spinIncrement, int spinDecrement){
sc=spinCount;
si=spinIncrement;
sd=spinDecrement;
} /**
* @description: 获取锁,有以下情况:(非公共平模式,不判断队伍存在与否)
* 1.当前锁为自由状态,如果cas成功,那么线程成功持有,否则失败
* 2.当前锁被人持有,如果持有人正是当前线程,那么获取成功,线程重入,否则失败
* @param
* @return: boolean 获取成功为true,失败为false
* @date: 2020/2/5
*/
public boolean aquire(){
if(state.get()==0){//这里不直接使用cas,是为了减少cas的调用,因为这是一条cpu指令,耗费资源较大(能省则省)
if(state.compareAndSet(0,1)) {
onwThread=Thread.currentThread();
return true;
}else return false;
}else if(onwThread==Thread.currentThread()) {
state.incrementAndGet();//重入
return true;
}else return false;
} /**
* @description: 自旋,如果成功,那么设置当前线程,增加下次自旋次数,否则减少下次自旋次数
* @param
* @return: boolean 自旋成功为true,失败为false
* @date: 2020/2/5
*/
public boolean spin(){
int temp=spinCount.get();//缓存本次自旋次数
while(temp>0) {
temp--;
if(state.compareAndSet(0,1)) {//自旋成功,设置当前线程,增加下次自旋次数
onwThread=Thread.currentThread();
spinCount.set(spinCount.get()+spinIncrement.get());
return true;
}
}
spinCount.set(Math.max(spinCount.get()-spinDecrement.get(),0));//自旋失败,减少下次自旋次数。
return false;
} /**
* @description: 并发入队,在入队前做最后一次询问,若成功入队,线程设置中断后,park,
* 醒来之后中断取反,再去尝试获取锁
* @param
* @return: void
* @date: 2020/2/5
*/
public void inQueue(){
if(!aquire()){//入队前的最后一次询问
concurrentLinkedQueue.add(Thread.currentThread());
System.out.println(Thread.currentThread().getName()+"入队等候");
Thread.currentThread().interrupt();//设置中断
LockSupport.park();
System.out.println(Thread.currentThread().getName()+"被唤醒");
Thread.currentThread().interrupt();//再次中断取反
tryAcquire();//醒来之后再去尝试
}else System.out.println(Thread.currentThread().getName()+"入队前最后一次获取成功");
} /**
* @description: state-1,如果此时state=0而且队列不为空,则唤醒队头,让其重新竞争
* @param
* @return: void
* @date: 2020/2/5
*/
public void release() throws RuntimeException{
if(onwThread==Thread.currentThread()) {
//这里先减-1再去唤醒,此时若有新线程进来则刚好非公平地获取到
if(state.decrementAndGet()==0&&!concurrentLinkedQueue.isEmpty()) {
Thread thread=concurrentLinkedQueue.poll();
LockSupport.unpark(thread);
}
if(state.get()==0) System.out.println(Thread.currentThread().getName()+"释放了锁");
else System.out.println(Thread.currentThread().getName()+"还要释放了"+state.get()+"次");
}
else throw new RuntimeException("非法释放,当前线程并非锁的拥有者!");
} /**
* @description: 线程首次Lock前调用,初始化个线程的自旋次数,增量,减量
* @param
* @return: void
* @date: 2020/2/5
*/
public void init(){
this.spinCount.set(sc);
this.spinIncrement.set(si);
this.spinDecrement.set(sd);
} /**
* @description: 尝试获取锁,失败后尝试自旋获取,若仍然失败则进入阻塞队列,park,等待别人用完后来unpark,
* 然后再次尝试获取锁。
* @param
* @return: void
* @date: 2020/2/5
*/
public void tryAcquire(){
//if(!aquire()&&!spin()) inQueue();
if(aquire()) {
if(state.get()==1) System.out.println(Thread.currentThread().getName()+"获取成功");
else System.out.println(Thread.currentThread().getName()+"获取成功,重入了"+(state.get()-1)+"次");
}
else{
System.out.println(Thread.currentThread().getName()+"获取失败,自旋"+spinCount.get()+"次");
if(spinCount.get()>0&&spin()) System.out.println(Thread.currentThread().getName()+"自旋成功");
else{
System.out.println(Thread.currentThread().getName()+"自旋失败,尝试进入队列");
inQueue();
}
}
} }
public class MyLock implements Lock{
private QueueSynchronizer queueSynchronizer =new QueueSynchronizer();
public void init() { queueSynchronizer.init();}
@Override
public void lock() {
queueSynchronizer.tryAcquire();
}
@Override
public void unlock() {
queueSynchronizer.release();
}
}
public interface Lock {
void lock();
void unlock();
}
public class MyLockTest {
private static int sum=100;
static MyLock myLock = new MyLock();
public static void main(String[] args) {
for (int j = 0; j < 10; j++) {
new Thread(()->{
myLock.init();//各线程初次使用时先初始化
myLock.lock();
myLock.lock();
for (int i = 0; i < 10; i++) {
System.out.println(--sum+" "+Thread.currentThread().getName()+"one");
}
myLock.unlock();
for (int i = 0; i < 10; i++) {
System.out.println(--sum+" "+Thread.currentThread().getName()+"two");
}
myLock.unlock();
}).start();
}
}
}
利用Atomic, ThreadLocal, 模仿AQS, ReentrantLock的更多相关文章
- paip.提升性能----java 无锁结构(CAS, Atomic, Threadlocal, volatile, 函数式编码, 不变对象)
paip.提升性能----java 无锁结构(CAS, Atomic, Threadlocal, volatile, 函数式编码, 不变对象) 1 锁的缺点 2 CAS(Compare ...
- JUC AQS ReentrantLock源码分析
警告⚠️:本文耗时很长,先做好心理准备,建议PC端浏览器浏览效果更佳. Java的内置锁一直都是备受争议的,在JDK1.6之前,synchronized这个重量级锁其性能一直都是较为低下,虽然在1.6 ...
- 技本功丨利用 Atomic 构建 React 项目工作流,so easy!
近日刷微博,#2018年结婚率创新低#荣登热门话题NO.1,沪浙最不积极. 生活压力越大,缺爱的人也越来越多...据本萌的不完全观察,程序猿虽然是压力加成的职业,在袋鼠云还是有不少早早脱了单.至于,脱 ...
- 利用art.template模仿VUE 一次渲染多个模版
TypeScript代码 import template = require('art-template/lib/template-web'); interface TemplateBindConfi ...
- 利用art.template模仿VUE
首先先看一下Typescript代码: import template = require('art-template/lib/template-web'); interface TemplateBi ...
- 扒一扒ReentrantLock以及AQS实现原理
提到JAVA加锁,我们通常会想到synchronized关键字或者是Java Concurrent Util(后面简称JCU)包下面的Lock,今天就来扒一扒Lock是如何实现的,比如我们可以先提出一 ...
- 深圳某小公司面试题:AQS是什么?公平锁和非公平锁?ReentrantLock?
AQS总体来说没有想象中那么难,只要了解它的实现框架,那理解起来就不是什么问题了. AQS在Java还是占很重要的地位的,面试也是经常会问. 目前已经连载11篇啦!进度是一周更新两篇,欢迎持续关注 [ ...
- ReentrantLock 实现原理
使用 synchronize 来做同步处理时,锁的获取和释放都是隐式的,实现的原理是通过编译后加上不同的机器指令来实现. 而 ReentrantLock 就是一个普通的类,它是基于 AQS(Abstr ...
- JAVA并发-同步器AQS
什么是AQS aqs全称为AbstractQueuedSynchronizer,它提供了一个FIFO队列,可以看成是一个用来实现同步锁以及其他涉及到同步功能的核心组件,常见的有:ReentrantLo ...
随机推荐
- $NOIp$做题记录
虽然去年做了挺多了也写了篇一句话题解了但一年过去也忘得差不多了$kk$ 所以重新来整理下$kk$ $2018(4/6$ [X]积木大赛 大概讲下$O(n)$的数学方法. 我是从分治类比来的$QwQ$. ...
- 洛谷$P4045\ [JSOI2009]$密码 $dp$+$AC$自动机
正解:$dp$+$AC$自动机+搜索 解题报告: 传送门$QwQ$ 首先显然先建个$AC$自动机,然后考虑设$f_{i,j,k}$表示长度为$i$,现在在$AC$自动机的第$j$个位置,已经表示出来的 ...
- 1026 程序运行时间 (15 分)C语言
题目描述 要获得一个C语言程序的运行时间,常用的方法是调用头文件time.h,其中提供了clock()函数,可以捕捉从程序开始运行到clock()被调用时所耗费的时间.这个时间单位是clock tic ...
- BZOJ 2038: [2009国家集训队]小Z的袜子 (莫队)
题目传送门:小Z的袜子 Description 作为一个生活散漫的人,小Z每天早上都要耗费很久从一堆五颜六色的袜子中找出一双来穿.终于有一天,小Z再也无法忍受这恼人的找袜子过程,于是他决定听天由命…… ...
- 基于Arduino的按键控制LED实验
I/O 口的意思即为INPUT 接口和OUTPUT 接口,到目前为止我们设计的小灯实验都还只是应用到Arduino 的I/O 口的输出功能,这个实验我们来尝试一下使用Arduino的I/O 口的输入功 ...
- [apue] 作为 daemon, 启动 Unix Domain Socket 侦听失败?
前段时间写一个传递文件句柄的小 demo,有 server 端.有 client 端,之间通过 Unix Domain Socket 通讯. 在普通模式下,双方可以正常建立连接,当server端作为d ...
- 一文搞定Spring Boot + Vue 项目在Linux Mysql环境的部署(强烈建议收藏)
本文介绍Spring Boot.Vue .Vue Element编写的项目,在Linux下的部署,系统采用Mysql数据库.按照本文进行项目部署,不迷路. 1. 前言 典型的软件开发,经过" ...
- IDEA不编译空文件夹
今天做项目的时候发现idea编译工程不会编译空文件夹,在resources下新建了个存储文件的空文件夹,编译后target里竟然没有,一直报空指针. 随便丢一个文件进去就行了,放一个demo.txt的 ...
- CocoaPods中的头文件import导入时不能自动补齐的解决方法
1.选择target(就是左边你的工程target)-->BuildSettings-->search Paths下的User Header Search Paths 2.添加“$(POD ...
- spring boot 的中文乱码
首先 自检IDEA的编码 配置文件加入设置http tomcat spring.http.encoding.force=true spring.http.encoding.charset=UTF-8 ...