什么是原子类,有什么作用

① 不可分割

② 一个操作是不可中断的,即便是多线程的情况下也可以保证

③ java.util.concurrent.atomic

④ 原子类的作用和锁类似,是为了保证并发情况下的线程安全。不过原子类相对于锁有一点的优势

  • 粒度更细:原子变量可以把竞争范围缩小到变量级别,这是我们可以获得的最细的粒度的情况了,通常锁的粒度都要大于原子变量的粒度

  • 效率更高:通常,使用原子类的效率会比使用锁的效率更高,除了高度竞争的情况

6类原子类纵览

Atomic*基本类型原子类,以AtomicInteger为例

1.AtomicInteger方法

2.常用方法

public final int get()//获取当前值

public final int getAndSet(int newValue)//获取当前值,并设置新的值

pubilic final int getAndIncrement()//获取当前值,并自增

pubilic final int getAndDecrement()//获取当前值,并自增

public final int getAndAdd(int delta)//获取当前值,并加上预期值

boolean compareAndSet(int expect,int update)//如果输入的值等于预期值,则以原子方式将该值设置为输入值(update)

/*** 演示AtomicInteger的基本用法,对比原子类的线程安全问题,使用了原子类以后,不需要加锁,也可以保证线程安全问题*/
public class AtomicIntegerDemo1 implements Runnable  {private static final AtomicInteger atomicInteger = new AtomicInteger();public void incrementAtomic(){atomicInteger.getAndIncrement();}private static volatile int basicCount = 0;public /*synchronized*/ void incrementBasic(){//synchronized保证线程安全basicCount++;}public static void main(String[] args) throws InterruptedException {AtomicIntegerDemo1 r = new AtomicIntegerDemo1();Thread thread1 = new Thread(r);Thread thread2 = new Thread(r);thread1.start();thread2.start();thread1.join();thread2.join();System.out.println("原子类的结果是 "+atomicInteger.get());System.out.println("普通变量的结果是 "+basicCount);}@Overridepublic void run() {for (int i = 0; i < 10000; i++) {incrementAtomic();incrementBasic();}}
}

Atomic*Arrray数组类型原子类

/*** 演示原子数组的使用方法*/
public class AtomicArrayDemo {public static void main(String[] args) {AtomicIntegerArray atomicIntegerArray = new AtomicIntegerArray(1000);Incrementer incrementer = new Incrementer(atomicIntegerArray);Decrementer decrementer = new Decrementer(atomicIntegerArray);Thread[] threadsIncrementer = new Thread[100];Thread[] threadsDecrementer = new Thread[100];for (int i = 0; i < 100; i++) {threadsIncrementer[i] = new Thread(incrementer);threadsDecrementer[i] = new Thread(decrementer);threadsIncrementer[i].start();threadsDecrementer[i].start();}for (int i = 0; i < 100; i++) {try {threadsDecrementer[i].join();threadsIncrementer[i].join();} catch (InterruptedException e) {e.printStackTrace();}}for (int i = 0; i < atomicIntegerArray.length(); i++) {//            if (atomicIntegerArray.get(i) != 0){//                System.out.println("发现了错误"+i);
//            }System.out.println(atomicIntegerArray.get(i));}System.out.println("运行结束");}
}class Decrementer implements Runnable {private AtomicIntegerArray array;public Decrementer(AtomicIntegerArray array) {this.array = array;}@Overridepublic void run() {for (int i = 0; i < array.length(); i++) {array.getAndDecrement(i);}}
}class Incrementer implements Runnable {private AtomicIntegerArray array;public Incrementer(AtomicIntegerArray array) {this.array = array;}@Overridepublic void run() {for (int i = 0; i < array.length(); i++) {array.getAndIncrement(i);}}
}

Atomic*Reference引用类型原子类

自旋锁使用

public class SpinLock {private AtomicReference<Thread> sign = new AtomicReference<>();public void lock() {Thread current = Thread.currentThread();while (!sign.compareAndSet(null, current)) {System.out.println("自旋获取失败,再次尝试");}}public void unlock() {Thread current = Thread.currentThread();sign.compareAndSet(current, null);}public static void main(String[] args) {SpinLock spinLock = new SpinLock();Runnable runnable = new Runnable() {@Overridepublic void run() {System.out.println(Thread.currentThread().getName() + "开始尝试获取自旋锁");spinLock.lock();System.out.println(Thread.currentThread().getName() + "获取到了自旋锁");try {Thread.sleep(300);} catch (InterruptedException e) {e.printStackTrace();} finally {spinLock.unlock();System.out.println(Thread.currentThread().getName() + "释放了自旋锁");}}};Thread thread1 = new Thread(runnable);Thread thread2 = new Thread(runnable);thread1.start();thread2.start();}
}

把普通变量升级为原子类

用AtomicIntegerFieldUpdate升级原有变量

public class AtomicIntegerFieldUpdaterDemo implements Runnable {static Candidate tom;static Candidate peter;public static AtomicIntegerFieldUpdater<Candidate> scoreUpdater =AtomicIntegerFieldUpdater.newUpdater(Candidate.class, "score");@Overridepublic void run() {for (int i = 0; i < 10000; i++) {peter.score++;scoreUpdater.getAndIncrement(tom);}}public static class Candidate {volatile int score;//不支持private、static}public static void main(String[] args) throws InterruptedException {tom = new Candidate();peter = new Candidate();AtomicIntegerFieldUpdaterDemo r = new AtomicIntegerFieldUpdaterDemo();Thread thread1 = new Thread(r);Thread thread2 = new Thread(r);thread1.start();thread2.start();thread1.join();thread2.join();System.out.println("普通的变量:" + peter.score);System.out.println("升级后的结果:" + tom.score);}
}

注意点:

① 可见范围:变量不能为private

② 不支持static

Adder累加器

① java8引入的,相比较是一个比较新的类

② 高并发下LogAdder比AtomicLog效率高,不过本质是空间换时间

③ 竞争激烈的时候,LongAdder把不同线程对应到不同的Cell上进行修改,降低了冲突的概率,是多段锁的理念,提高了并发性

④ LongAdder适合的场景是统计求和计数的场景,而且LongAdder基本只提供了add方法,而AtomicLong还具有cas方法

演示多线程下AtomicLong的性能,有20个线程对同一个AtomicLong累加(由于竞争很激烈,每一次加法,都要flush和refresh,导致很耗费资源)

/*** 演示高并发场景下,LongAdder比AtomicLong性能好*/
@SuppressWarnings("all")
public class AtomicLongDemo {public static void main(String[] args) throws InterruptedException {AtomicLong counter = new AtomicLong();ExecutorService service = Executors.newFixedThreadPool(20);long start = System.currentTimeMillis();for (int i = 0; i < 10000; i++) {service.submit(new Task(counter));}service.shutdown();while (!service.isTerminated()) {}long end = System.currentTimeMillis();System.out.println(counter.get());System.out.println("AtomicLon耗时:" + (end - start));}private static class Task implements Runnable {private AtomicLong counter;public Task(AtomicLong counter) {this.counter = counter;}@Overridepublic void run() {for (int i = 0; i < 10000; i++) {counter.incrementAndGet();}}}
}

LongAdder演示

@SuppressWarnings("all")
public class AtomicAdderDemo {public static void main(String[] args) throws InterruptedException {LongAdder counter = new LongAdder();ExecutorService service = Executors.newFixedThreadPool(20);long start = System.currentTimeMillis();for (int i = 0; i < 10000; i++) {service.submit(new Task(counter));}service.shutdown();while (!service.isTerminated()) {}long end = System.currentTimeMillis();System.out.println(counter.sum());System.out.println("LongAdder耗时:"+(end-start));}private static class Task implements Runnable {private LongAdder counter;public Task(LongAdder counter) {this.counter = counter;}@Overridepublic void run() {for (int i = 0; i < 10000; i++) {counter.increment();}}}
}

累加结果耗时显示

LongAdder改进和原理:

① 在内部,这个LongAdder的实现原理和刚才的AtomicLong是不同的,刚才的AtomicLong的实现原理是,每一次加法都需要做同步,所以在高并发的时候会导致冲突比较多,也就降低了效率

② 而此时的LongAdder,每个线程都有一个计数器,仅用来在自己的线程内计算,这样一来就不会和其他线程的计数器干扰

③ 如上图,第一个线程的计数器的数值,也就是ctr’,为1的时候,可能线程2的计数器ctr’’的数值已经是3了,他们之间并不存在竞争关系,所以在加和的过程中,根本不需要同步机制,也不需要刚才的flush和reflush。这里没有一个公共的counter来给所有线程统一计数

④ LongAdder引入了分段累加的概念,内部有一个base变量和一个Cell[]数组共同参与计数:

  • base变量:竞争不激烈,直接累加到该变量上

  • Cell[]:竞争激烈,各个线程分散累加到自己的槽Cell[i]中,

⑤ sum源码

 public long sum() {Cell[] as = cells; Cell a;long sum = base;if (as != null) {for (int i = 0; i < as.length; ++i) {if ((a = as[i]) != null)sum += a.value;}}return sum;}

Accumulator累加器

public class LongAccumulatorDemo {public static void main(String[] args) {LongAccumulator accumulator = new LongAccumulator((x,y)->x+y,0);ExecutorService executor = Executors.newFixedThreadPool(8);IntStream.range(1,10).forEach(i->executor.submit(()->accumulator.accumulate(i)));executor.shutdown();while (!executor.isTerminated()){}System.out.println(accumulator.getThenReset());}
}

使用场景:

① 适用于需要大量计算,并且需要并行计算的场景,如果不需要并行计算,可用for循环解决问题,用了Accumulator累加器可利用多核同时计算,提供效率

② 计算的顺序不能成为瓶颈,线程1可能在线程5之后运行,也可能在之前运行,不影响最终结果

什么是原子类,原子类有那些,有什么作用相关推荐

  1. 【java】抽象类下有两个具体子类,子类下有两个实例

    抽象类下有两个具体子类,子类下有两个实例 package p1; import java.text.*;public class Java_2 {public static void main(Str ...

  2. 不花一分钱让手机支持原声原唱

    原声原唱是目前新手机的一大卖点,我的三星X199没有这个功能,但通过刷机,我让它拥有了这些功能,真是爽呆了,不花一分钱耶. 买手机之后不久就有了数据线,但一直只是通过它传一些20K以下的铃声.50K以 ...

  3. python父类和子类_python子类父类

    1.类的定义 代码如下: 复制代码复制代码 !/usr/bin/env python coding:utf8 class Hotel(object): """docstr ...

  4. 父类指针指向子类对象,子类指针不能指向父类对象

    class Parent { public:int a };class Child :public Parent { public:int b; }Parent类占内存大小范围:int a; 占4个字 ...

  5. 为什么父类引用可以指向子类对象 子类引用不能指向父类对象 泛型

    假设有父类Fu ,其子类为Zi ,从对象的内存角度,假设Fu类里面的变量占内存2M, Zi 类里的变量占内存1M: Fu f = new Fu ();//系统将分配2M内存 Zi z = new Zi ...

  6. c语言实现补码转换成原码(原->补同理)

    说明:         反码=原码符号位不变,其余按位取反:         补码=原码符号位不变,其余按位取反,再加1:         补码的补码就是原码: 补码和原码相互转换,代码实现: /* ...

  7. g楦和h楦的区别_原楦鞋与正品区别-原楦原纸板是什么意思-莆田原楦版本什么意思...

    原版原楦什幺意思 楦 xuàn 做鞋用的模型.本作[楥](篆书)从木,从爰.[形声]后作[楦].从木,宣声. 本义;使鞋子定型的模具.楦子|鞋楦 鞋子,楦内长是什幺意思? 鞋的内里长度 请问文中&qu ...

  8. java子类的子类_java中的子类是什么

    java中的子类是什么 发布时间:2020-06-26 15:57:41 来源:亿速云 阅读:152 作者:Leah java中的子类是什么?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编 ...

  9. Flutter - sqflite(原滋原味)

    原文地址 #Installing(安装) Use this package as a library Depend on it Add this to your package's pubspec.y ...

  10. 最近更换系统盘踩了不少坑,主要是软件运行错误,造成原成原系统盘被损毁

    更换系统盘应该注意的问题: PE系统要是经过测试的系统,这次损毁的PE我很信任,但是有个克隆软件,今天一用就损毁了,他的名字是"Acronis True Image",不建议用他克 ...

最新文章

  1. python32位安装
  2. python进阶-第二季第一章
  3. 通过COM组件在Web上实现Kinect骨骼追踪、声控截屏保存的功能
  4. LeetCode 1753. 移除石子的最大得分(优先队列)
  5. 微信小程序黑客马拉松即将开始,来做最酷的 Mini Program Creators! 1
  6. 信息系统开发平台OpenExpressApp -如何部署OEA应用
  7. 鸿蒙系统的平板电脑,亓纪的想法 篇三百六十三:骁龙870+鸿蒙2.0!首款鸿蒙系统平板曝光,支持第二代M-Pencil...
  8. MySQL配置与启动
  9. 模糊数学与matlab
  10. 391 卡信乐卡盟程序
  11. 数学建模清风微信公众号的习题答案(基础篇-操作题)
  12. php 字符相似度比较,php 比较两个字符串的相似度
  13. 华为手机的拨号键有大用处,不仅能用来打电话,还有4个隐藏功能
  14. web开发第三方登陆之facebook登陆
  15. 麻将判断胡牌 java_麻将基本胡的算法——Java
  16. openstack 单元测试助手tox
  17. cordic ip核 vivado_Xilinx Vivado CORDIC IP核求解atan 反正切
  18. Android Pitfall - 扒一扒RadioGroup 和 RadioButton
  19. vue2 mapEchart示例
  20. 希尔伯特的23个数学问题

热门文章

  1. 基于PKI密钥认证在CentOS7.9搭建OpenVPN(一)
  2. 孩子不听话怎么办,能打吗?
  3. 如何把扫描pdf转换成txt
  4. android am发送广播,adb shell am broadcast 手动发送广播及adb shell am/pm其他命令
  5. 国产手机的几种品牌发展模式
  6. 格兰杰因果关系检验(Granger Causality Test)
  7. Web前端技术课程设计——技术栈【SpringBoot+Vue+MySQL+MyBatis】的在线英语考试系统
  8. 双操作系统安装(七)Windows及Archlinux双系统安装教程
  9. h3c linux驱动 wn612_H3C iNode智能客户端安装指导(Linux)-7.3-5PW102
  10. 【合规性检查-Fitness】基于完全解析的拟合度评估方法