i++的操作不是原子的,因为它不会作为一个不可分割的操作来执行。它实际包含了三个独立的操作,读取i的值,将值加1,然后将计算结果写入i。这是一个读取—修改—写入的操作序列,并且其结果状态依赖于之前的状态。
volatile的语义不足以确保递增操作的原子性,除非你能确保只有一个线程对变量执行写操作。
1、可以通过使用线程安全类,如
private final AtomicInterger i = new AtomicInteger(0);
i.incrementAndGet();
2、当然也可以通过synchronized和Lock来保证其原子性。