阿里推荐 LongAdder ,不推荐 AtomicLong !

简介: 在分布式系统中,计数器是一个常见的需求。为了实现高并发、高可用的计数器,我们需要选择一个合适的实现方式。在 Java 中,有两种常见的计数器实现方式:AtomicLong 和 LongAdder。最近,阿里巴巴在一份技术报告中推荐使用 LongAdder ,而不是 AtomicLong。本文将介绍这两种计数器的原理和优缺点,并分析为什么阿里巴巴推荐使用 LongAdder。

 其他系列文章导航

Java基础合集

数据结构与算法合集

设计模式合集

多线程合集

分布式合集

ES合集


文章目录

其他系列文章导航

文章目录

前言

一、CAS

1.1 CAS 全称

1.2 通俗理解CAS

1.3 CAS的问题

1.4 解决 ABA 问题

二、LongAdder

2.1 什么是 LongAdder

2.2 为什么推荐推荐 LongAdder

三、AtomicLong

3.1 什么是 AtomicLong

3.2 为什么不推荐 AtomicLong

四、总结


前言

在分布式系统中,计数器是一个常见的需求。为了实现高并发、高可用的计数器,我们需要选择一个合适的实现方式。

在 Java 中,有两种常见的计数器实现方式:AtomicLong 和 LongAdder。

最近,阿里巴巴在一份技术报告中推荐使用 LongAdder ,而不是 AtomicLong 。

本文将介绍这两种计数器的原理和优缺点,并分析为什么阿里巴巴推荐使用 LongAdder 。


一、CAS

1.1 CAS 全称

全称:compare and swap,比较并交换。

虽然翻译过来是[比较并交换],但它是一个原子性的操作,对应到CPU指令为 cmpxchg 。

image.gif编辑

1.2 通俗理解CAS

    1. CAS 有三个操作数:当前值A、内存值V、要修改的新值B。
    2. 假设 当前值A 跟 内存值V 相等,那就将内存值V 改成B。
    3. 假设 当前值A 跟 内存值V 不相等,要么就重试,要么就放弃更新。
    4. 将当前值与内存值进行对比,判断是否有被修改过,这就是CAS的核心。

    1.3 CAS的问题

    CAS有个缺点就是会带来 ABA 的问题。

    从CAS更新的时候,我们可以发现它只比对当前值和内存值是否相等,这会带来个问题,下面我举例说明下:

      1. 假设线程A读到当前值是10,可能线程B把值修改为100,然后线程C又把值修改为10。
      2. 等到线程A拿到执行权时,因为当前值和内存值是一致的,线程A是可以修改的!
      3. 站在线程A的角度来说,这个值是从未被修改的 。
      4. 这是不合理的,因为我们从上帝的角度来看,这个变量已经被线程B和线程C修改过了。

      1.4 解决 ABA 问题

      要解决ABA的问题,Java 也提供了 AtomicStampedReference 类供我们用,说白了就是加了个版本,比对的就是内存值+版本是否一致。

      疑问:

      为什么阿里巴巴开发手册提及到推荐使用 LongAdder 对象,比AtomicLong 性能更好(减少乐观锁的重试次数)?

      原因:

      因为 AtomicLong 做累加的时候实际上就是多个线程操作同一个目标资源。

      在高并发时,只有一个线程是执行成功的,其他的线程都会失败,不断自旋(重试),自旋会成为瓶颈。

      而 LongAdder 的思想就是把要操作的目标资源 分散,到数组 Cell 中。

      每个线程对自己的 Cell 变量的 value 进行原子操作,大大降低了失败的次数。

      这就是为什么在高并发场景下,推荐使用 LongAdder  的原因。


      二、LongAdder

      2.1 什么是 LongAdder

      LongAdder是JDK1.8由Doug Lea大神新增的原子操作类,位于java.util.concurrent.atomic包下,LongAdder在高并发的场景下会比AtomicLong 具有更好的性能,代价是消耗更多的内存空间

      LongAdder是Google开源的一个高性能计数器实现。它采用了一种分段锁的策略,将一个long型的变量分割成多个16字节的段,每个段都使用一个独立的AtomicLong进行更新。这样,在高并发场景下,多个线程可以同时对不同的段进行更新操作,互不干扰。

      LongAdder的优点是并发性能高,适用于高并发的场景。由于采用了分段锁的策略,LongAdder可以避免AtomicLong中的竞争问题。此外,LongAdder还支持可扩展性,可以通过增加更多的段来提高性能。但是,LongAdder的缺点是代码相对复杂一些,需要更多的维护成本。

      2.2 为什么推荐推荐 LongAdder

      LongAdder设计思想上,采用分段的方式降低并发冲突的概率。通过维护一个基准值base和 Cell 数组

      如下图所示:

      image.gif编辑


      三、AtomicLong

      3.1 什么是 AtomicLong

      AtomicLong是Java提供的一个原子类,用于实现高并发的计数器。它利用了CAS(Compare-and-Swap)操作来保证线程安全。在AtomicLong中,每次计数操作都会先读取当前值,然后使用CAS操作更新值。如果值没有被其他线程修改过,则更新成功,否则需要重新尝试。

      AtomicLong的优点是简单易用,性能也不错。但是,在高并发场景下,AtomicLong可能会出现竞争问题。因为多个线程可能同时读取和更新同一个AtomicLong的当前值,导致数据不一致。此外,AtomicLong的CAS操作也可能因为硬件和操作系统的原因出现失败的情况。

      3.2 为什么不推荐 AtomicLong

      在LongAdder之前,当我们在进行计数统计的时,通常会使用AtomicLong来实现。AtomicLong能保证并发情况下计数的准确性,其内部通过CAS来解决并发安全性的问题。

      如下图所示:

      image.gif编辑

      图里可以看出在高并发情况下,当有大量线程同时去更新一个变量,任意一个时间点只有一个线程能够成功,绝大部分的线程在尝试更新失败后,会通过自旋的方式再次进行尝试,这样严重占用了CPU的时间片,进而导致系统性能问题。


      四、总结

      阿里巴巴推荐使用LongAdder的原因主要有以下几点:

        1. 高并发性能:LongAdder采用分段锁的策略,可以避免AtomicLong中的竞争问题,提高并发性能。在分布式系统中,高并发性能是非常重要的。
        2. 可扩展性:LongAdder支持可扩展性,可以通过增加更多的段来提高性能。这对于需要处理大量请求的分布式系统来说是非常有利的。
        3. 代码简单易懂:虽然LongAdder的代码相对复杂一些,但是相对于AtomicLong来说更容易理解和维护。这对于开发人员来说是非常重要的。
        4. 更好的适用场景:阿里巴巴推荐使用LongAdder主要是因为在分布式系统中需要一个高性能、高可用的计数器实现。而LongAdder正好符合这个需求。

        总之,阿里巴巴推荐使用LongAdder的原因主要是因为它的高并发性能、可扩展性、代码简单易懂以及更好的适用场景。当然,在实际应用中还需要根据具体场景和需求进行选择和优化。

        目录
        相关文章
        |
        24天前
        |
        存储 安全 Java
        ConcurrentLinkedQueue详解
        通过本文的介绍,希望您能够深入理解 `ConcurrentLinkedQueue`的工作原理、主要特性、常用方法以及实际应用,并在实际开发中灵活运用这些知识,编写出高效、健壮的并发程序。
        25 3
        |
        8月前
        |
        安全 Java
        AtomicInteger 的使用
        AtomicInteger 的使用
        60 1
        |
        8月前
        |
        缓存 安全 算法
        Java并发基础:原子类之AtomicInteger全面解析
        【2月更文挑战第2天】AtomicInteger类提供了线程安全的整数操作,它通过利用底层硬件的原子性指令,能够在多线程环境中高效地实现整数的无锁更新,避免了传统同步机制带来的性能开销,在高并发场景下成为计数器可大幅提高程序的执行效率,同时又保证了数据一致性。
        283 16
        Java并发基础:原子类之AtomicInteger全面解析
        |
        缓存 Java 索引
        源码系列-LongAdder和AtomicLong对比
        源码系列-LongAdder和AtomicLong对比
        45 0
        |
        算法 Java
        【JUC基础】05. Synchronized和ReentrantLock
        前面两篇中分别讲了Synchronized和ReentrantLock。两种方式都能实现同步锁,且也都能解决多线程的并发问题。那么这两个有什么区别呢? 这个也是一个高频的面经题。
        115 0
        AtomicInteger.getAndIncrement()怎么保证线程安全(四)
        问题 我们知道i++线程不安全,那AtomicInteger.getAndIncrement()怎么保证线程安全
        235 0
        AtomicInteger.getAndIncrement()怎么保证线程安全(四)
        |
        缓存 安全 Java
        JUC - BlockingQueue
        JUC - BlockingQueue
        136 0
        JUC - BlockingQueue
        |
        安全 Oracle Java
        阿里为什么推荐使用LongAdder,而不是volatile?
        阿里为什么推荐使用LongAdder,而不是volatile?
        159 0
        阿里为什么推荐使用LongAdder,而不是volatile?
        |
        安全 算法 Java
        详解java并发原子类AtomicInteger(基于jdk1.8源码分析)
        java并发包里面的类一直是学习和面试的重点,这篇文章主要是对java并发包的其中一个类AtomicInteger的讲解。从为什么要出现AtomicInteger再到其底层原理来一个分析。
        345 0
        详解java并发原子类AtomicInteger(基于jdk1.8源码分析)
        |
        安全 Java 开发者
        ThreadLocalRandom 是线程安全的吗?
        前言 最近在写一些业务代码时遇到一个需要产生随机数的场景,这时自然想到 jdk 包里的 Random 类。
        155 0
        ThreadLocalRandom 是线程安全的吗?

        热门文章

        最新文章

        下一篇
        开通oss服务