Atomic
CAS原语
public final boolean compareAndSet(int expect, int update) {
return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
}
在 x86 下的指令CMPXCHG实现了CAS,前置LOCK既可以达到原子性操作。截止2013,大部分多核处理器均支持CAS。
ABA 问题
public final int getAndSet(int newValue) {
for (;;) {
//先读取val=1
int current = get();
//线程被抢占
if (compareAndSet(current, newValue))
return current;
}
}
thread2 修改val=3,又修改回1。
thread1继续执行,发现期望值与“原值”(其实被修改过了)相同,完成CAS操作。
AtomicIntegerFieldUpdater<T>
AtomicLongFieldUpdater<T>
AtomicReferenceFieldUpdater<T,V>
是基于反射的原子更新字段的值。
相应的API也是非常简单的,但是也是有一些约束的。
(1)字段必须是volatile类型的!volatile到底是个什么东西。
(2)字段的描述类型(修饰符public/protected/default/private)是与调用者与操作对象字段的关系一致。也就是说调用者能够直接操作对象字段,那么就可以反射进行原子操作。但是对于父类的字段,子类是不能直接操作的,尽管子类可以访问父类的字段。
(3)只能是实例变量,不能是类变量,也就是说不能加static关键字。
(4)只能是可修改变量,不能使final变量,因为final的语义就是不可修改。实际上final的语义和volatile是有冲突的,这两个关键字不能同时存在。
(5)对于AtomicIntegerFieldUpdater和AtomicLongFieldUpdater只能修改int/long类型的字段,不能修改其包装类型(Integer/Long)。如果要修改包装类型就需要使用AtomicReferenceFieldUpdater。
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater; public class AtomicIntegerFieldUpdaterDemo { class DemoData{
public volatile int value1 = 1;
volatile int value2 = 2;
protected volatile int value3 = 3;
private volatile int value4 = 4;
}
AtomicIntegerFieldUpdater<DemoData> getUpdater(String fieldName) {
return AtomicIntegerFieldUpdater.newUpdater(DemoData.class, fieldName);
}
void doit() {
DemoData data = new DemoData();
System.out.println("1 ==> "+getUpdater("value1").getAndSet(data, 10));
System.out.println("3 ==> "+getUpdater("value2").incrementAndGet(data));
System.out.println("2 ==> "+getUpdater("value3").decrementAndGet(data));
System.out.println("true ==> "+getUpdater("value4").compareAndSet(data, 4, 5));
}
public static void main(String[] args) {
AtomicIntegerFieldUpdaterDemo demo = new AtomicIntegerFieldUpdaterDemo();
demo.doit();
}
}
在上面的例子中DemoData的字段value3/value4对于AtomicIntegerFieldUpdaterDemo类是不可见的,因此通过反射是不能直接修改其值的。
AtomicMarkableReference类描述的一个<Object,Boolean>的对,可以原子的修改Object或者Boolean的值,这种数据结构在一些缓存或者状态描述中比较有用。这种结构在单个或者同时修改Object/Boolean的时候能够有效的提高吞吐量。
AtomicStampedReference类维护带有整数“标志”的对象引用,可以用原子方式对其进行更新。对比AtomicMarkableReference类的<Object,Boolean>,AtomicStampedReference维护的是一种类似<Object,int>的数据结构,其实就是对对象(引用)的一个并发计数。但是与AtomicInteger不同的是,此数据结构可以携带一个对象引用(Object),并且能够对此对象和计数同时进行原子操作。
最新文章
- nginx服务傻瓜搭建
- 如何设置文本文件的默认保存编码为UTF-8
- Python 基礎 - 列表的使用
- FFmpeg - 音频解码过程
- 【转】NGUI创建UIRoot后报NullReferenceException的解决办法
- 当今app行业 比较流行的 简称 汇总
- Android(java)学习笔记193:利用谷歌API对数据库增删改查(推荐使用)
- C#中byte[]与string的转换
- oracle通过query导出指定条件的数据
- Volley该框架使用了大量的请求图片
- 随机获取部分List<;Object>;集合
- photosho 等距复制或旋转复制
- [Artoolkit] Can I Use LGPL code for commercial application
- Python3.6学习笔记(二)
- 【WPF】生成二维码
- CSS布局奇淫技巧之--各种居中<;转>;
- 爬虫——请求库之selenium模块
- openURL的使用
- python版本selenium定位方式(不止八种哦)
- File Searching
热门文章
- 在Windows的Tomcat环境下部署Solr 4.7.0
- 修改织梦默认提示";dedecms提示信息!";
- NGUI 新版操作教程
- cocos2d回忆
- FastJSON 简介及其Map/JSON/String 互转
- DrawText
- twisted udp编程
- Python: 什么是*args和**kwargs
- git无法clone远程代码库及git代理设置
- delphi提示“Undeclared_identifier”的缺少引用单元列表