java并发编程-12个原子类
背景
多线程更新变量的值,可能得不到预期的值,当然增加syncronized关键字可以解决线程并发的问题。
这里提供另外一种解决问题的方案,即位于 java.util.concurrent.atomic包下的原子操作类,提供了一种用法简单,性能高效,线程安全的更新变量的方式。

其它两个附带的类顺带看了一下:
LongAddr 多线程先的sum操作
LongAccomulator 多线程下的函数式操作,性能低于AtomicLong,主要是函数式的支持;
简单分类:

基本类型原子类
使用原子的方式更新基本类型,包括:
- AtomicBoolean
- AtomicInteger
- AtomicLong
核心方法:
直接看源码了。

类签名:
public class AtomicInteger extends Number implements java.io.Serializable {}
| 方法 | 功能说明 |
|---|---|
| 构造方法 | 两个构造方法,不传或者传入值 |
| get方法 | get()获取值;对应的有set(int)方法,layzySet(int) 懒设置 |
| getAndAdd(int) | 获得老值然后增加一个数字, 对应的有addAndGet(int)增加一个数字并返回新值 |
| getAndSet(int) | 获得老值然后更新为新值 |
| getAndIncreament() | 获得老值然后+1,对应的有increamentAndGet() +1然后返回新值 |
| getAndDecrement() | 获得老值然后-1 ,对应的有decrementAndGet() -1然后返回新值 |
| getAndUpdate(IntUnaryOperator) | 获取老值然后执行一个函数得到新值并设置,对应的有updateAndGet(IntUnaryOperator) 先执行内置函数式接口再返回新值 |
| getAndAccumulate(int,IntBinaryOperator) | 获取老值,然后把老值和第一个参数进行函数运算的返回值并设置 ,对应的有accumulateAndGet(int,IntBinaryOperator) 执行运算然后返回新值 |
| compareAndSet(int,int) | 对比如果跟预期值相等则设置为新值,对应的有weakCompareAndSet(int,int)这个是不保证顺序设置 |
| toString | 返回数字的字符串形式 |
| number继承过来的方法 | longValue(),byteValue()直接做了类型转换 |
| object继承过来的方法 | 直接沿用Object的方法 |
底层是基于 unsafe来实现,基于CAS来原子性;
来研究一下unsafe的实现源码:
/**
* Atomically decrements by one the current value.
*
* @return the previous value
*/
public final int getAndDecrement() {
return unsafe.getAndAddInt(this, valueOffset, -1);
}
public final int getAndAddInt(Object var1, long var2, int var4) {
int var5;
do {
var5 = this.getIntVolatile(var1, var2);
} while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));
return var5;
}
大致的处理流程是:死循环,对比; 也就是CAS;
利用了volatile的特性,多线程的变量可见性;
原子数组
通过原子的方式更新数组中的某个元素;
包含3个类:
- AtomicIntegerArray
- AtomicIntegerLongArray
- AtomicReferenceArray
抓一个类来分析研究一下:
public class AtomicIntegerArray implements java.io.Serializable {}

| 方法 | 说明 |
|---|---|
| 构造方法 | public AtomicIntegerArray(int length),public AtomicIntegerArray(int[] array)这里会做一个clone,不影响传入的数组的值 |
| length | 得到内部数组的长度 |
| get,set,layziset | 获取,设置,懒设置 |
| compareAndSet,weakCompareAndSet | CAS操作, weak方法不保证操作的顺序性 |
| getAndAdd,getAndUpdate,getAndAccumulate | 有反向的方法,就是先计算,然后返回新值 |
| toString | 打印出数组【数字1,数字2】 |
原子类型的操作比较特殊一点:
/**
* Atomically adds the given value to the element at index {@code i}.
*
* @param i the index
* @param delta the value to add
* @return the previous value
*/
public final int getAndAdd(int i, int delta) {
return unsafe.getAndAddInt(array, checkedByteOffset(i), delta);
}
private long checkedByteOffset(int i) {
if (i < 0 || i >= array.length)
throw new IndexOutOfBoundsException("index " + i);
return byteOffset(i);
}
private static long byteOffset(int i) {
return ((long) i << shift) + base;
}
这块获取数组中的值时候用到了一个移位操作;
更新引用
原子更新类AtomicInterger只能更新一个变量,如果要更新多个不同的变量就要用到原子更新引用 类型提供的类;
- AtomicReference 更新引用类型
- AtomicReferenceFieldUpdater 更新引用类型的字段
- AtomicMarkableRerence 更新带有标志位的引用类型
以AtomicReference为例子:
签名:public class AtomicReference implements java.io.Serializable{}
方法:

| 方法 | 说明 |
|---|---|
| 构造方法 | public AtomicReference(V initialValue) 带初始值;public AtomicReference() |
| get,set,lazySet | 设置,获取,懒设置 |
| compareAndSet,weakCompareAndSet | CAS操作,weak方法不保证顺序 |
| getAndSet,getAndUpdate,getAndAccumulate | 有反向的操作 |
| toString | 打印出里面的对象 |
底层分析:
/**
* Atomically sets to the given value and returns the old value.
*
* @param newValue the new value
* @return the previous value
*/
@SuppressWarnings("unchecked")
public final V getAndSet(V newValue) {
return (V)unsafe.getAndSetObject(this, valueOffset, newValue);
}
利用了unsafe提供的特性保证了原子操作;
原子更新字段
需要原子的更新某个类的某个字段,需要用到原子更新字段类;
- AtomicIntegerFiledUpdater 不用多说,原子更新类的Interger字段
- AtomicLongFieldUpdater 不用多说,原子更新类的Long字段
- AtomicStampedReference 原子更新带版本号的引用类型,可以原子的更新引用和引用的版本号,解决ABA问题;
使用要点:
- 每次必须使用静态方法 newUpdater创建一个更新器,设置类和属性;
- 更新的类的属性必须使用 public volatile修饰;
package com.cocurrenttest.atomictest;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
/**
* 说明:人实体
* @author carter
* 创建时间: 2019年12月06日 19:27
**/
@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class Person {
private String name;
//注意,只能是int,Integer会报错哦
public volatile int age;
}
package com.cocurrenttest.atomictest;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
/**
* 说明:TODO
* @author carter
* 创建时间: 2019年12月06日 19:26
**/
public class TestAtomicIntegerUpdater {
public static void main(String[] args) {
final AtomicIntegerFieldUpdater<Person> personAtomicIntegerFieldUpdater = AtomicIntegerFieldUpdater.newUpdater(Person.class, "age");
Person person = Person.builder().name("lifuchun").age(30).build();
personAtomicIntegerFieldUpdater.addAndGet(person,1);
final int age = personAtomicIntegerFieldUpdater.get(person);
System.out.println(age);
assert age==31 : "更新失败";
}
}
小结
原子操作类都介绍了一下,适当的场景的话,简单的说一下我使用过的两个场景:
- 多任务去数据copy的时候的对账,计算出总的修改行数,或者迁移的订单总金额,方便两边对比;
- 流式操作的lambda表达式里面需要传入的局部变量为final,但是 一般的类型在方法中不是final的,还需要在中间过程中修改,IDE提示可以使用原子类包装,然后带上final去修改;

package com.cocurrenttest.atomictest;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;
import java.util.stream.Collectors;
/**
* 说明:TODO
* @author carter
* 创建时间: 2019年12月06日 19:36
**/
public class TestStream {
public static void main(String[] args) {
Integer age = 25 ;
final String name="b";
//some condition to change
name ="bbb";
final List<Person> personList = Arrays.asList(
Person.builder().name("aaa").age(10).build(),
Person.builder().name("bbb").age(20).build(),
Person.builder().name("ccc").age(30).build()
)
.stream()
.filter(item -> item.getAge() >= age)
.filter(item->item.getName().contains(name))
.collect(Collectors.toList());
System.out.println(personList);
}
public static void main2(String[] args) {
Integer age = 25 ;
final AtomicReference<String> name=new AtomicReference<>("b");
//some condition to change
name.set("bbb");;
final List<Person> personList = Arrays.asList(
Person.builder().name("aaa").age(10).build(),
Person.builder().name("bbb").age(20).build(),
Person.builder().name("ccc").age(30).build()
)
.stream()
.filter(item -> item.getAge() >= age)
.filter(item->item.getName().contains(name.get()))
.collect(Collectors.toList());
System.out.println(personList);
}
}
原创不易,转载请注明出处。
java并发编程-12个原子类的更多相关文章
- 【Java并发工具类】原子类
前言 为保证计数器中count=+1的原子性,我们在前面使用的都是synchronized互斥锁方案,加锁独占访问的方式未免太过霸道,于是我们来介绍另一种解决原子性问题的无锁方案:原子变量.在正式介绍 ...
- 【Java_多线程并发编程】JUC原子类——原子类中的volatile变量和CAS函数
JUC中的原子类是依靠volatile变量和Unsafe类中的CAS函数实现的. 1. volatile变量的特性 内存可见性(当一个线程修改volatile变量的值后,另一个线程就可以实时看到此变量 ...
- 【Java_多线程并发编程】JUC原子类——AtomicLong原子类
1. AtomicLong是基本原子类中的一种 AtomicLong是对长整形进行原子操作. 1.1 AtomicLong类的函数列表 // 构造函数 AtomicLong() // 创建值为init ...
- 【Java_多线程并发编程】JUC原子类——4种原子类
根据修改的数据类型,可以将JUC包中的原子操作类可以分为4种,分别是: 1. 基本类型: AtomicInteger, AtomicLong, AtomicBoolean ;2. 数组类型: Atom ...
- Java并发编程原理与实战十三:JDK提供的原子类原理与使用
原子更新基本类型 原子更新数组 原子更新抽象类型 原子更新字段 原子更新基本类型: package com.roocon.thread.t8; import java.util.concurren ...
- 12、Java并发编程:阻塞队列
Java并发编程:阻塞队列 在前面几篇文章中,我们讨论了同步容器(Hashtable.Vector),也讨论了并发容器(ConcurrentHashMap.CopyOnWriteArrayList), ...
- Java并发编程75道面试题及答案
1.在java中守护线程和本地线程区别? java中的线程分为两种:守护线程(Daemon)和用户线程(User). 任何线程都可以设置为守护线程和用户线程,通过方法Thread.setDaemon( ...
- Java并发编程75个问答
1.在java中守护线程和本地线程区别? java中的线程分为两种:守护线程(Daemon)和用户线程(User). 任何线程都可以设置为守护线程和用户线程,通过方法Thread.setDaemon( ...
- Java并发编程73道面试题及答案
原文出处:https://blog.csdn.net/qq_34039315/article/details/7854931 1.在java中守护线程和本地线程区别? java中的线程分为两种:守护线 ...
随机推荐
- LInux内核配置过程
内核版本 linux 2.6.32.2 配置内核的过程 配置内核可以通过执行 make menuconfig 来进行,下面分析该命令的执行流程 执行该目标 %config: scripts_basic ...
- [NOIP模拟]文本编辑器 题解
bsoj5089 文本编辑器 /* 题意描述 九发明了一个完美的文本编辑器.这个编辑器拥有两个光标(cursor),所以九能够同时在两处地方插入和删除文本.这个编辑器除了正常的编辑功能以外,还有一些只 ...
- 备份下ESP8266的AT指令集手册和用例手册中文版,准备为V7做几个ESP8266的例子
指令集手册:https://files.cnblogs.com/files/armfly/4a-esp8266_at_instruction_set_cn.rar 用例手册: https://file ...
- ETCD:客户端v3
原文地址:etcd/clientv3 etcd/clientv3是v3版本的Go etcd官方客户端 安装 go get go.etcd.io/etcd/clientv3 开始 创建客户端使用clie ...
- 转载 SAP用户权限控制设置及开发
创建用户SU01 事务码:SU01,用户主数据的维护,可以创建.修改.删除.锁定.解锁.修改密码等 缺省:可以设置用户的起始菜单.登录的默认语言.数字显示格式.以及日期和时间的格式设置 参数:SAP很 ...
- ubuntu安装navicat
ubuntu下安装navicat1.官网下载https://www.navicat.com.cn/download/navicat-premium,不清楚系统是32位的还是64位的,可以用”uname ...
- JS查找某个字符在字符串中出现的位置及次数
var str = 'fdhfgcsaedvcfhgfh'; var index = str.indexOf('f'); // 字符出现的位置 var num = 0; // 这个字符出现的次数 wh ...
- C lang:Pointer and multidimensional array
Xx_Introduction Double indrection:Address of Address;Pointer of Pointer Ax_Code #include<stdio.h& ...
- Android 项目优化(一):项目代码规范优化
项目代码规范为主要包含:类,常量,变量,ID等命名规范:注释规范:分包规范:代码风格规范. 项目代码规范是软件开发过程中非常重要的优化环节. 目前的开发社区提供了很多的开发规范文档,阿里巴巴推出了&l ...
- Android UI开发之开源控件项目整理
一.Banner 1.https://github.com/youth5201314/banner Android广告图片轮播控件,支持无限循环和多种主题,可以灵活设置轮播样式.动画.轮播和切换时间. ...