首页 > 编程语言 >面试必备:Java JUC AtomicLong 实现解析

面试必备:Java JUC AtomicLong 实现解析

时间:2022-10-17 18:07:47浏览次数:84  
标签:JUC Object Java AtomicLong long offset HotSpotIntrinsicCandidate public



基于OpenJDK 12


本文的目的是为后续文章解析LongAdder做一个引子,以便两者对比。

Atomic Package解析参考(比如lazySet原理解析):

​[译]Java Concurrent Atomic Package详解​

AtomicLong的常用方法如下:

  • long addAndGet(long delta):以原子方式将输入的数值与实例中的值(AtomicLong里的value)相加,并返回结果。
  • compareAndSet(long expectedValue, long newValue):如果输入的数值等于预期值,则以原子方式将该值设置为输入的值。
  • long getAndIncrement():以原子方式将当前值加1,注意,这里返回的是自增前的值。
  • void lazySet(long newValue):最终会设置成newValue,使用lazySet设置值后,可能导致其他线程在之后的一小段时间内还是可以读到旧的值。
  • long getAndSet(long newValue):以原子方式设置为newValue的值,并返回旧值。

AtomicLong示例代码如下:

public class AtomicLongTest {    
static AtomicLong ai = new AtomicLong(1);
public static void main(String[] args) {
System.out.println(ai.getAndIncrement());
System.out.println(ai.get());
}
}

输出结果如下:

1  
2

那么getAndIncrement是如何实现原子操作的呢?让我们一起分析其实现原理,getAndIncrement的源码如下:

public final long getAndIncrement() {   
return U.getAndAddLong(this, VALUE, 1L);
}
@HotSpotIntrinsicCandidate
public final long getAndAddLong(Object o, long offset, long delta) {
long v;
do {
v = getLongVolatile(o, offset);
} while (!weakCompareAndSetLong(o, offset, v, v + delta));
return v;
}
/** Volatile version of {@link #getLong(Object, long)} */
@HotSpotIntrinsicCandidate
public native long getLongVolatile(Object o, long offset);


@HotSpotIntrinsicCandidate
public final boolean weakCompareAndSetLong(Object o, long offset,
long expected,
ong x) {
return compareAndSetLong(o, offset, expected, x);
}
/**
* Atomically updates Java variable to {@code x} if it is currently
* holding {@code expected}.
*
* <p>This operation has memory semantics of a {@code volatile} read
* and write. Corresponds to C11 atomic_compare_exchange_strong.
*
* @return {@code true} if successful
*/
@HotSpotIntrinsicCandidate
public final native boolean compareAndSetLong(Object o, long offset,
long expected,
long x);


@HotSpotIntrinsicCandidate JDK的源码中,被@HotSpotIntrinsicCandidate标注的方法,在HotSpot中都有一套高效的实现,该高效实现基于CPU指令,运行时,HotSpot维护的高效实现会替代JDK的源码实现,从而获得更高的效率。

源码getAndAddLong(Object o, long offset, long delta)中do while循环体是实现的关键所在,其逻辑是:

  • 第一步先取得AtomicLong里存储的数值,
  • 第二步对AtomicLong的当前数值进行加1操作,
  • 第三步调用weakCompareAndSetLong方法来进行原子更新操作,该方法先检查当前数值是否等于v,等于意味着AtomicLong的值没有被其他线程修改过,则将weakCompareAndSetLong的当前数值更新成v+delta的值,如果不等于v,weakCompareAndSetLong方法会返回false,程序会进入do while循环重新进行weakCompareAndSetLong操作。

这里隐含了一个问题,当对于共享变量(假设变量名字是a)的竞争非常激烈的时候,在当前线程读取a、改变a之间,a的值会被别的线程改变,从而导致当前线程一直重试(自旋),一直占用CPU。

这就引出另一个问题,对于锁抢占很激烈的时候,串行是最好的解决办法。比如使用synchronized。


java.util.concurrent.atomic中的原子操作基本是基于Unsafe或者VarHandle实现的。

源代码:

​https://github.com/jiankunking/openjdk12/blob/master/src/java.base/share/classes/java/util/concurrent/atomic/AtomicLong.java​


本文参考 《Java并发编程的艺术》 作者:方腾飞 魏鹏 程晓明

< END >

喜欢就转发个朋友圈呗

面试必备:Java JUC AtomicLong 实现解析_JUC

衣舞晨风

标签:JUC,Object,Java,AtomicLong,long,offset,HotSpotIntrinsicCandidate,public
From: https://blog.51cto.com/jiankunking/5763772

相关文章

  • Java实现支付宝扫码支付沙箱环境实例
    1、进入蚂蚁金服开放平台登录注册网址为:https://openhome.alipay.com/developmentDocument.htm进入后登录自己的支付宝账户,登录后点击进入管理中心  进入后,下拉到最......
  • 1110 区块反转(JAVA)
    给定一个单链表L,我们将每K个结点看成一个区块(链表最后若不足K个结点,也看成一个区块),请编写程序将L中所有区块的链接反转。例如:给定L为1→2→3→4→5→6→7→8,K为......
  • Java 多线程(八)同步方法及同步块
    同步方法由于我们可以通过private关键字来保证数据对象只能被方法访问,所以我们需要针对方法提出一套机制,这套机制就是synchronized关键字,它包括两种用法:synchronized方......
  • 后缀表达式 (又称 逆波兰表达式) Java代码的简单实现
    表达式求值问题好久没有发随笔了,最近学习复习数据结构的时候看到了后缀表达式(逆波兰表达式)发现了栈的精巧,自己想实现一下,本来想用C写的,但是实在太困难了,所有写了个简单......
  • JavaWeb(一):MySql基础
    目录​​1、数据库相关概念​​​​1.1数据库​​​​1.2数据库管理系统​​​​1.3常见的数据库管理系统​​​​1.4SQL​​​​2、MySQL​​​​2.1MySQL安装​​​......
  • 一篇文章让你搞懂Java中的静态代理和动态代理
    什么是代理模式代理模式是常用的java设计模式,在Java中我们通常会通过new一个对象再调用其对应的方法来访问我们需要的服务。代理模式则是通过创建代理类(proxy)的方式间......
  • Java实现Excel和Office Open XML之间的相互转换
    前言OfficeOpenXML(也被称为OOXML)是一种压缩的、基于XML的Excel、Word和演示文档格式。有时,你可能需要将Excel文件转换为OfficeOpenXML,以使其在各种应用程序和平台上可......
  • Java使用RestTemplate发送Post请求时携带参数
    Stringurl="https://www.baidu.com";HttpHeadersheaders=newHttpHeaders();//设置请求头,自己从浏览器复制一个,如果请求的网站没要求也可以不设置headers.set("us......
  • Java 多线程(七)三大不安全案例
    一,买票//不安全买票publicclassUnsafeBuyTickets{publicstaticvoidmain(String[]args){BuyTicketsbuyTickets=newBuyTickets();new......
  • Java NIO——缓冲区Buffer
    基本介绍缓冲区(Buffer):缓冲区本质上是一个可以读写数据的内存块,可以理解成是一个容器对象(含数组),该对象提供了一组方法,可以更轻松地使用内存块,缓冲区对象内置了一些机制,能......