java并发编程之原子类

当多个线程同时更新公共变量,会导致线程不安全,通常大家可以会想到使用synchronized关键字或者Lock来解决这个问题,synchronized和Lock可以保证多个线程不会同时更新该公共变量。为了使用更简单,性能更高效,jdk1.5提出原子操作类。

原子操作类主要集中在Atomic(java.util.concurrent.atomic)包下,按照原子更新方式,这些原子操作类大致可以分为四种:原子更新基本类型、原子更新数组、原子更新引用以及原子更新属性,接下来就这四种类型原子操作类的具体实现做相关分析。

原子更新基本类型

Atomic包主要提供三个类来更新基本类型变量:

  1. AtomicBoolean:用来更新布尔型变量;

  2. AtomicInteger:用来更新整型变量;

  3. AtomicLong:用来更新长整型变量。

接下来就以AtomicLong为切入点,分析一下具体的源码实现。

AtomicLong源码分析

AtomicLong有以下比较常用的方法:

  • compareAndSet方法


    compareAndSet实现

    调用Unsafe的compareAndSwapLong方法实现比较设置,如果当前value与预期值expect相等,则将value设置为update的值。

  • getAndSet方法


    getAndSet实现

    以原子方式设置value为newValue的值,并且返回更新前的value值。需要注意的是,使用compareAndSet方法进行原子操作时,如果当前value与current值不相等则意味着value被其他线程修改过,这时候compareAndSet方法会返回false,程序会再次进入循环重复进行compareAndSet操作直到操作成功。

  • add系列方法
    AtomicLong提供addAndGet方法和getAndAdd方法来做加法运算。


    add系列方法实现

    从源码可以看出addAndGet方法的返回值是value加上delta之后的值,而getAndAdd方法的返回值是做加法之前的value值,这也是这两个方法唯一的区别,大家在使用的时候需要注意。

  • increment系列方法
    同add一样,AtomicLong同样提供两个方法:incrementAndGet和getAndIncrement方法来做自增操作。


    increment系列方法实现

    同add的两个方法一样,incrementAndGet和getAndIncrement唯一的区别就在于前者返回值是自增以后的value值,而后者返回的是自增之前的value值。

  • decrement系列方法
    同样的,AtomicLong也提供两个方法:decrementAndGet和getAndDecrement方法来做自减操作。


    decrement系列方法实现

    同前add、increment一样,decrement的两个方法区别也是返回值不一样。

  • lazySet方法


    lazySet实现

    lazySet方法调用Unsafe的putOrderedLong方法将value设置成newValue,但是使用lazySet设置值可能会导致其他线程在之后的一小段时间内读到的值还是设置之前的旧值,至于为什么会出现延迟,会在后续文章分析Hotspot的unsafe具体实现中给出。

到这里为止,原子更新基本类型的相关源码分析就告一段落了,总体来说,Atomic包下提供的三个更新基本类型变量的原子类源码大致一致,有兴趣的读者可以自行阅读。最后,再强调一点,AtomicBoolean的value的类型并不是布尔型,而是整型,它会把对应的布尔值转换成整型,true的时候对应1,false的时候对应0。至于为什么是整型,是因为Unsafe提供的compareAndSwap方法不能修改布尔型变量:


compareAndSwap系列方法
案例1
  • 需求:实现自增计数器,要求线程安全。

  • 代码实现

    代码实现demo

    运行结果:
    运行结果

    1000个线程操作自增之后计数器结果为1000,线程安全。

原子更新数组

Atomic包提供三个类来以原子的方式更新数组里的元素:

  1. AtomicIntegerArray:用来更新整型数组里的元素;

  2. AtomicLongArray:用来更新长整型数组里的元素;

  3. AtomicReferenceArray:用来更新引用类型数组里的元素。

接下来还是以AtomicLongArray为例,分析具体的源码实现。

AtomicLongArray源码分析

AtomicLongArray同AtomicLong对外提供的方法大致一致,只不过前者是操作数组,后者是操作基本类型。

  • 成员变量


    成员变量

    AtomicLongArray自己维护一个长整型数组array,对数组元素的操作实质是对array的操作。

  • 构造方法


    构造方法

    AtomicLongArray提供两个构造方法,从入参为长整型数组的构造方法可以看出,AtomicLongArray会将传入数组clone一份,所以AtomicLongArray对内部的数组元素进行修改时不会影响传入的数组,一定要注意这点!!!

  • CAS实现
    其实Atomic包下系列方法实现基本都差不太多,在这里就只着重分析一下基础的CAS的具体实现:


    CAS实现

    从源码可以看出,AtomicLongArray的CAS依赖于Unsafe的compareAndSwapLong方法,同样需要注意的是,getAndSet方法同样也是死循环,直到value更新成功,到这里基本上可以得出一个结论:Atomic包下的CAS基本上都是配合死循环来实现的。

原子更新引用类型

原子更新基本类型每次只能更新一个变量,假如需要更新多个变量怎么办呢?针对这个问题,Atomic包提供引用类型类来一次更新多个变量:

  1. AtomicReference:用于更新引用类型,可以理解为更新Object;

  2. AtomicMarkableReference:用于更新带有标记位的引用类型;

  3. AtomicStampedReference:用于更新带有版本号的引用类型,该类将版本号与引用类型关联起来,可以解决使用CAS进行原子更新时可能会出现的ABA问题。

接下来以AtomicReference为例,分析一下具体的源码实现。

AtomicReference源码分析

AtomicReference实质是被用来更新Object。

  • 成员变量及构造方法


    成员变量及构造方法

    从源码可以看出,AtomicReference自己维护一个泛型对象value,同时提供两个构造方法,带参数的构造方法可以初始化value为传入的对象。

  • CAS实现
    同样看一下CAS的基本实现,看看AtomicReference怎么来做到原子更新Object的:


    CAS实现

    从源码可以看出,AtomicReference的CAS实现主要依赖于Unsafe的compareAndSwapObject方法。

源码分析到这里基础的东西基本就结束了,可能大家还是一脸懵逼,实现这么简单,那到底怎么使用呢,接下来就给大家一个简单的demo,看看它具体是怎么用的。

案例2
  • 需求:使用AtomicReference原子更新用户信息;

  • 代码实现

    代码实现demo

    运行结果:
    运行结果

原子更新属性

AtomicReference系列可以更新Object,同样的,针对Object的属性,Atomic提供一下方法来更新Object的属性:

  1. AtomicIntegerFieldUpdater:用于更新Object的整型属性;

  2. AtomicLongFieldUpdater:用于更新Object的长整型属性;

  3. AtomicReferenceFieldUpdater:用于更新Object的引用类型属性。

以AtomicIntegerFieldUpdater为例,分析一下源码的具体实现。

AtomicIntegerFieldUpdater源码分析

在使用AtomicIntegerFieldUpdater来更改Object整型属性大致分为两步:

  1. 使用静态方法newUpdater创建一个更新器,设置需要更新的类和属性;

  2. 调用相关CAS系列方法更新属性,需要注意的是,更新的属性必须使用public volatile修饰

接下来就先从如何创建一个更新器开始,分析具体的源码实现。

  • 创建更新器


    创建更新器

    AtomicIntegerFieldUpdater是抽象类,它提供静态方法newUpdater来创建更新器。从源码可以看出,newUpdater方法实际上是创建了一个AtomicIntegerFieldUpdater的子类的对象,接下来就去看一下这个子类AtomicIntegerFieldUpdaterImpl的构造方法主要做了什么。

    AtomicIntegerFieldUpdaterImpl构造方法

    构造方法

    整个构造方法主要干了这些事:

    1. 利用反射获取到Object指定(fieldName)属性;

    2. 调用ensureMemberAccess方法校验该属性是否允许外部访问,这一步校验限制了该属性必须是public的;

    3. 校验当前属性是否是int型,并且被volatile关键字修饰;

    4. 初始化私有属性cclass,tclass和offset。

创建完更新器之后,我们来看看如何更改指定属性的值。

  • CAS更改属性值
    AtomicIntegerFieldUpdater同AtomicInteger等原子更新基本类型一样,提供一些列方法来更新属性值,这里只给出基本的compartAndSet方法和调用compartAndSet方法实现属性值的更新的getAndSet方法实现。

    • compartAndSet方法实现


      compareAndSet方法实现

      从源码实现可以看出,compareAndSet方法依赖Unsafe的compareAndSwapInt方法实现原子更新int型属性。

    • getAndSet方法实现


      getAndSet方法实现

      同AtomicInteger的getAndSet方法实现一样的逻辑,循环调用compareAndSet方法直到属性值更新成功。

    AtomicIntegerFieldUpdater的add、increment和decrement系列方法实现均调用compareAndSet方法实现,实现的逻辑与原子更新基本类型的几个类实现一致,在这里就不再做详细讲解,有兴趣的读者下来可以自行阅读源码。接下来,还是一样给大家一个简单的AtomicIntegerFieldUpdater使用demo,看看具体怎么来用它。

案例3
  • 需求:原子更新用户信息的年龄信息;

  • 代码实现

    代码实现

    运行结果:


    运行结果
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容