volatile的应用
volatile的两条实现原则:
1)Lock前缀指令会引起处理器缓存回写到内存。
2)一个处理器的缓存回写到内存会导致其他处理器的缓存无效。
synchronized的的实现原理和应用
- 重量级锁
·对于普通同步方法,锁是当前实例对象。
·对于静态同步方法,锁是当前类的Class对象。
·对于同步方法块,锁是Synchonized括号里配置的对象。
monitorenter指令是在编译后插入到同步代码块的开始位置,而monitorexit是插入到方法结束处和异常处,JVM要保证每个monitorenter必须有对应的monitorexit与之配对。
synchronized用的锁是存在Java对象头里的。如果对象是数组类型,则虚拟机用3个字宽(Word)存储对象头,如果对象是非数组类型,则用2字宽存储对象头。
- 锁的升级与对比
锁一共有4种状态,级别从低到高依次是:无锁状态、偏向锁状态、轻量级锁状态和重量级锁状态。锁可以升级但不能降级,意味着偏向锁升级成轻量级锁后不能降级成偏向锁。这种锁升级却不能降级的策略,目的是为了提高获得锁和释放锁的效率。
①偏向锁:大多数情况下,锁不仅不存在多线程竞争,而且总是由同一线程多次获得,为了让线程获得锁的代价更低而引入了偏向锁。当一个线程访问同步块并获取锁时,会在对象头和栈帧中的锁记录里存储锁偏向的线程ID,以后该线程在进入和退出同步块时不需要进行CAS操作来加锁和解锁,只需简单地测试一下对象头的Mark Word里是否存储着指向当前线程的偏向锁。如果测试成功,表示线程已经获得了锁。如果测试失败,则需要再测试一下Mark Word中偏向锁的标识是否设置成1(表示当前是偏向锁):如果没有设置,则使用CAS竞争锁;如果设置了,则尝试使用CAS将对象头的偏向锁指向当前线程。
②轻量级锁:线程在执行同步块之前,JVM会先在当前线程的栈桢中创建用于存储锁记录的空间,并将对象头中的Mark Word复制到锁记录中,官方称为Displaced Mark Word。然后线程尝试使用CAS将对象头中的Mark Word替换为指向锁记录的指针。如果成功,当前线程获得锁,如果失败,表示其他线程竞争锁,当前线程便尝试使用自旋来获取锁。轻量级解锁时,会使用原子的CAS操作将Displaced Mark Word替换回到对象头,如果成功,则表示没有竞争发生。如果失败,表示当前锁存在竞争,锁就会膨胀成重量级锁。
③锁的优缺点对比
锁 | 优点 | 缺点 | 适用场景 |
---|---|---|---|
偏向锁 | 加锁和解锁不需要额外的消耗 | 如果线程间存在锁竞争,会带来额外的锁撤销的消耗 | 适用于只有一个线程访问同步块场景 |
轻量级锁 | 竞争的线程不会阻塞,提高了程序的响应速度 | 如果始终得不到锁竞争的线程,使用自旋会消耗CPU | 追求响应速度 |
重量级锁 | 线程竞争不使用自旋,不会消耗CPU | 线程阻塞,响应时间缓慢 | 追求吞吐量 |
原子操作的实现原理
-
处理器实现原子操作
处理器提供总线锁定和缓存锁定两个机制来保证复杂内存操作的原子性。
但是有两种情况下处理器不会使用缓存锁定:①当操作的数据不能被缓存在处理器内部,或操作的数据跨多个缓存行(cache line)时,则处理器会调用总线锁定;②有些处理器不支持缓存锁定。 -
Java实现原子操作
在Java中可以通过锁和循环CAS的方式来实现原子操作。
①使用循环CAS实现原子操作
public class Counter {
private AtomicInteger ai = new AtomicInteger(0);
private int i = 0;
/**
* 非线程安全计数器
* */
private void count(){
i++;
}
/**
* 使用CAS实现线程安全计数器
* */
private void safeCount(){
for(;;){
int j = ai.get();
boolean ca = ai.compareAndSet(j, ++j);
if(ca){
break;
}
}
}
public static void main(String[] args) throws Exception {
final Counter counter = new Counter();
List<Thread> list = new ArrayList<>();
for (int i = 0; i < 100; i++) {
//并发实现原子操作
Thread t = new Thread(new Runnable() {
@Override
public void run() {
for (int j = 0; j < 1000; j++) {
counter.count();
counter.safeCount();
}
}
});
list.add(t);
}
//启动所有线程
for (Thread thread : list) {
thread.start();
}
//等待所有线程执行完成
for (Thread thread : list) {
thread.join();
}
System.out.println("非线程安全计算结果:"+counter.i);
System.out.println("线程安全计算结果:"+counter.ai.get());
}
}
存在问题:ABA问题,循环时间长开销大,以及只能保证一个共享变量的原子操作。
②使用锁机制实现原子操作
锁机制保证了只有获得锁的线程才能够操作锁定的内存区域。