J2SE 5.0提供了一组atomic class来帮助我们简化同步处理。基本工作原理是使用了同步synchronized的方法实现了对一个long, integer, 对象的增、减、赋值(更新)操作. 比如对于++运算符AtomicInteger可以将它持有的integer 能够atomic 地递增。在需要访问两个或两个以上 atomic变量的程序代码(或者是对单一的atomic变量执行两个或两个以上的操作)通常都需要被synchronize以便两者的操作能够被当作是一个atomic的单元。
对array atomic变量来说,一次只有一个索引变量可以变动,并没有功能可以对整个array做atomic化的变动。
关于Atomic的几个方法
getAndSet() : 设置新值,返回旧值.
compareAndSet(expectedValue, newValue) : 如果当前值(current value)等于期待的值(expectedValue), 则原子地更新指定值为新值(newValue), 如果更新成功,返回true, 否则返回false, 换句话可以这样说: 将原子变量设置为新的值, 但是如果从我上次看到的这个变量之后到现在被其他线程修改了(和我期望看到的值不符), 那么更新失败
从effective java (2)中拿来的一个关于AtomicReference的一个例子:
Java代码
-
public class AtomicTest {
-
private int x, y;
-
-
private enum State {
-
NEW, INITIALIZING, INITIALIZED
-
};
-
-
private final AtomicReference<State> init = new AtomicReference<State>(State.NEW);
-
-
public AtomicTest() {
-
}
-
-
public AtomicTest(int x, int y) {
-
initialize(x, y);
-
}
-
-
private void initialize(int x, int y) {
-
if (!init.compareAndSet(State.NEW, State.INITIALIZING)) {
-
throw new IllegalStateException("initialize is error");
-
}
-
this.x = x;
-
this.y = y;
-
init.set(State.INITIALIZED);
-
}
-
-
public int getX() {
-
checkInit();
-
return x;
-
}
-
-
public int getY() {
-
checkInit();
-
return y;
-
}
-
-
private void checkInit() {
-
if (init.get() == State.INITIALIZED) {
-
throw new IllegalStateException("uninitialized");
-
}
-
}
-
-
}
-
-
public class AtomicTest {
-
private int x, y;
-
-
private enum State {
-
NEW, INITIALIZING, INITIALIZED
-
};
-
-
private final AtomicReference<State> init = new AtomicReference<State>(State.NEW);
-
-
public AtomicTest() {
-
}
-
-
public AtomicTest(int x, int y) {
-
initialize(x, y);
-
}
-
-
private void initialize(int x, int y) {
-
if (!init.compareAndSet(State.NEW, State.INITIALIZING)) {
-
throw new IllegalStateException("initialize is error");
-
}
-
this.x = x;
-
this.y = y;
-
init.set(State.INITIALIZED);
-
}
-
-
public int getX() {
-
checkInit();
-
return x;
-
}
-
-
public int getY() {
-
checkInit();
-
return y;
-
}
-
-
private void checkInit() {
-
if (init.get() == State.INITIALIZED) {
-
throw new IllegalStateException("uninitialized");
-
}
-
}
-
- }
public class AtomicTest {
private int x, y;
private enum State {
NEW, INITIALIZING, INITIALIZED
};
private final AtomicReference<State> init = new AtomicReference<State>(State.NEW);
public AtomicTest() {
}
public AtomicTest(int x, int y) {
initialize(x, y);
}
private void initialize(int x, int y) {
if (!init.compareAndSet(State.NEW, State.INITIALIZING)) {
throw new IllegalStateException("initialize is error");
}
this.x = x;
this.y = y;
init.set(State.INITIALIZED);
}
public int getX() {
checkInit();
return x;
}
public int getY() {
checkInit();
return y;
}
private void checkInit() {
if (init.get() == State.INITIALIZED) {
throw new IllegalStateException("uninitialized");
}
}
}
public class AtomicTest {
private int x, y;
private enum State {
NEW, INITIALIZING, INITIALIZED
};
private final AtomicReference<State> init = new AtomicReference<State>(State.NEW);
public AtomicTest() {
}
public AtomicTest(int x, int y) {
initialize(x, y);
}
private void initialize(int x, int y) {
if (!init.compareAndSet(State.NEW, State.INITIALIZING)) {
throw new IllegalStateException("initialize is error");
}
this.x = x;
this.y = y;
init.set(State.INITIALIZED);
}
public int getX() {
checkInit();
return x;
}
public int getY() {
checkInit();
return y;
}
private void checkInit() {
if (init.get() == State.INITIALIZED) {
throw new IllegalStateException("uninitialized");
}
}
}
上面的例子比较容易懂, 不过貌似没什么价值, 而在实际的应用中, 我们一般采用下面的方式来使用atomic class:
Java代码
-
public class CounterTest {
-
AtomicInteger counter = new AtomicInteger(0);
-
-
public int count() {
-
int result;
-
boolean flag;
-
do {
-
result = counter.get();
-
// 断点
-
// 单线程下, compareAndSet返回永远为true,
-
// 多线程下, 在与result进行compare时, counter可能被其他线程set了新值, 这时需要重新再取一遍再比较,
-
// 如果还是没有拿到最新的值, 则一直循环下去, 直到拿到最新的那个值
-
flag = counter.compareAndSet(result, result + 1);
-
} while (!flag);
-
-
return result;
-
}
-
-
public static void main(String[] args) {
-
final CounterTest c = new CounterTest();
-
new Thread() {
-
@Override
-
public void run() {
-
c.count();
-
}
-
}.start();
-
-
new Thread() {
-
@Override
-
public void run() {
-
c.count();
-
}
-
}.start();
-
-
new Thread() {
-
@Override
-
public void run() {
-
c.count();
-
}
-
}.start();
-
}
-
}
-
-
public class CounterTest {
-
AtomicInteger counter = new AtomicInteger(0);
-
-
public int count() {
-
int result;
-
boolean flag;
-
do {
-
result = counter.get();
-
// 断点
-
// 单线程下, compareAndSet返回永远为true,
-
// 多线程下, 在与result进行compare时, counter可能被其他线程set了新值, 这时需要重新再取一遍再比较,
-
// 如果还是没有拿到最新的值, 则一直循环下去, 直到拿到最新的那个值
-
flag = counter.compareAndSet(result, result + 1);
-
} while (!flag);
-
-
return result;
-
}
-
-
public static void main(String[] args) {
-
final CounterTest c = new CounterTest();
-
new Thread() {
-
@Override
-
public void run() {
-
c.count();
-
}
-
}.start();
-
-
new Thread() {
-
@Override
-
public void run() {
-
c.count();
-
}
-
}.start();
-
-
new Thread() {
-
@Override
-
public void run() {
-
c.count();
-
}
-
}.start();
-
}
- }
public class CounterTest {
AtomicInteger counter = new AtomicInteger(0);
public int count() {
int result;
boolean flag;
do {
result = counter.get();
// 断点
// 单线程下, compareAndSet返回永远为true,
// 多线程下, 在与result进行compare时, counter可能被其他线程set了新值, 这时需要重新再取一遍再比较,
// 如果还是没有拿到最新的值, 则一直循环下去, 直到拿到最新的那个值
flag = counter.compareAndSet(result, result + 1);
} while (!flag);
return result;
}
public static void main(String[] args) {
final CounterTest c = new CounterTest();
new Thread() {
@Override
public void run() {
c.count();
}
}.start();
new Thread() {
@Override
public void run() {
c.count();
}
}.start();
new Thread() {
@Override
public void run() {
c.count();
}
}.start();
}
}
public class CounterTest {
AtomicInteger counter = new AtomicInteger(0);
public int count() {
int result;
boolean flag;
do {
result = counter.get();
// 断点
// 单线程下, compareAndSet返回永远为true,
// 多线程下, 在与result进行compare时, counter可能被其他线程set了新值, 这时需要重新再取一遍再比较,
// 如果还是没有拿到最新的值, 则一直循环下去, 直到拿到最新的那个值
flag = counter.compareAndSet(result, result + 1);
} while (!flag);
return result;
}
public static void main(String[] args) {
final CounterTest c = new CounterTest();
new Thread() {
@Override
public void run() {
c.count();
}
}.start();
new Thread() {
@Override
public void run() {
c.count();
}
}.start();
new Thread() {
@Override
public void run() {
c.count();
}
}.start();
}
}
类似i++这样的"读-改-写"复合操作(在一个操作序列中, 后一个操作依赖前一次操作的结果), 在多线程并发处理的时候会出现问题, 因为可能一个线程修改了变量, 而另一个线程没有察觉到这样变化, 当使用原子变量之后, 则将一系列的复合操作合并为一个原子操作,从而避免这种问题, i++=>i.incrementAndGet()
原子变量只能保证对一个变量的操作是原子的, 如果有多个原子变量之间存在依赖的复合操作, 也不可能是安全的, 另外一种情况是要将更多的复合操作作为一个原子操作, 则需要使用synchronized将要作为原子操作的语句包围起来. 因为涉及到可变的共享变量(类实例成员变量)才会涉及到同步, 否则不必使用synchronized
分享到:
相关推荐
5.1 AtomicInteger与AtomicLong 5.2 AtomicReference 第六章:线程池及其应用 6.1 ThreadPoolExecutor 6.2 ScheduledThreadPoolExecutor 第七章:Fork/Join框架 7.1 ForkJoinPool 等等
Java concurrency之AtomicLong原子类_动力节点Java学院整理,动力节点口口相传的Java黄埔军校
Java中提供了`java.util.concurrent.atomic`包,该包包含了一系列的原子类,如AtomicInteger、AtomicLong、AtomicReference等。这些类提供的原子操作主要基于硬件层面的CAS(Compare and Swap,比较并交换)指令来...
- `AtomicIntegerFieldUpdater`、`AtomicLongFieldUpdater`和`AtomicReferenceFieldUpdater`:动态生成特定类的某个字段的原子更新器,适用于无法直接访问的类的字段。 4. 原子类的工作原理: 原子类内部利用了...
1. java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue ...25. 原子性引用型 AtomicReference
为了解决这个问题,Java提供了`java.util.concurrent.atomic`包下的原子类,如AtomicInteger、AtomicLong和AtomicReference等。这些原子类的更新操作由Java内存模型直接保证原子性。此外,`synchronized`关键字的...
1. java.util.concurrent - Java 并发工具包 2. 阻塞队列 BlockingQueue 3. 数组阻塞队列 ArrayBlockingQueue 4. 延迟队列 DelayQueue 5. 链阻塞队列 LinkedBlockingQueue ...25. 原子性引用型 AtomicReference
1. AtomicInteger、AtomicLong、AtomicReference等,这些原子类提供了一些原子操作,可以用来实现原子性。 2. 这些原子类可以用来解决线程安全问题,例如,使用AtomicInteger可以实现原子性的自增操作。 五、Java中...
数组阻塞队列ArrayBlockingQueue,延迟队列DelayQueue, ...原子性长整型 AtomicLong,原子性引用型 AtomicReference 修改数据: 一 服务端修改数据: 一 文章知识点与官方知识档案匹配,可进一步学习相关知识
Java 原子类是Java并发编程中非常重要的工具,它们提供了一种高效且线程安全的方式来更新变量,尤其在处理共享数据时。在JDK 7及之前,共有12个原子类,而在JDK 8中,又新增了4个。下面,我们将详细介绍这些原子类...
这些类包括AtomicInteger、AtomicLong、AtomicReference等,它们都是通过底层的CAS操作来实现无锁线程安全访问的。例如,AtomicInteger类提供了如compareAndSet、getAndIncrement等原子操作方法,这些方法内部都使用...
为了确保数据在并发访问时的正确性,Java提供了一系列的原子类,包括`AtomicInteger`,它为整数类型的变量提供了线程安全的更新操作。这个话题将深入探讨`AtomicInteger`与普通`int`变量在多线程环境下进行递增操作...
在Java中,`java.util.concurrent.atomic`包提供了多种原子类,如AtomicInteger、AtomicLong等,这些类支持原子性的增加、减小、更新等操作,避免了显式的同步锁的使用,提高了并发性能。例如,AtomicInteger的...
原子类如AtomicInteger、AtomicLong和AtomicReference等提供了一种无锁的更新机制,保证了在高并发场景下的数据一致性。它们通过CAS(Compare and Swap)操作实现原子性,避免了synchronized关键字带来的开销,提高...
原子操作:concurrent包提供了一些原子操作类,如AtomicInteger、AtomicLong等,可以实现无锁的线程安全操作,避免了使用锁带来的性能损耗。 线程池:concurrent包提供了Executor框架,可以方便地创建和管理线程池,...
在Kotlinx.AtomicFu中,主要包含了一些原子类,如`AtomicInt`, `AtomicLong`, `AtomicReference`等,它们提供了原子性的增加、减少、比较并交换(compareAndSet)等操作。 1. **AtomicInt**: 这个类提供了对整型值...
文档重点介绍了基本类型原子更新类,如AtomicInteger、AtomicLong和AtomicBoolean。这类类能够以原子方式更新基本数据类型的值。文档中的示例代码展示了如何通过这些类来确保多线程环境下对共享变量的并发安全访问。...
Atomic类,如AtomicInteger、AtomicLong和AtomicReference,提供了在不使用锁的情况下实现原子更新的能力。这些类提供了原子操作,如incrementAndGet(),decrementAndGet(),compareAndSet()等,用于在高并发环境下...