锁可以保护代码路径以串行的方式访问共享状态的独占访问。共享状态包含原子变量和对象的共享状态。
对共享状态的保护有两种方式:加锁或者使用原子类。其中原子类是为了提高效率实现原子变量的并发安全访问实现。其实现原理是通过魔法类来实现的,如下图所示。
/**
* Atomically sets to the given value and returns the old value.
*
* @param newValue the new value
* @return the previous value
*/
public final int getAndSet(int newValue) {
return unsafe.getAndSetInt(this, valueOffset, newValue);
}
第二种就是使用锁机制来实现保护对象的共享状态
在复合操作执行的过程中持有一把锁,即将一些列复合操作封装在一个同步代码块内,但这样虽然可以使复合操作变成原子操作,但这里存在一个问题:就是所有访问该变量的所有位置上都必须使用同一把锁,严重降低了系统的性能。
而对象的内置锁和其状态没有内在的联系,在A线程获取与对象关联的锁时,并不能阻止其他线程访问该对象,只是阻止其他线程获取同一把锁。该模式是将所有的可变状态封装在对象中,并通过对象的内置锁对所有访问可变状态的代码进行同步,使得在该对象上不会发生并发访问。
不过需要注意的是并不是所有变量都需要加锁,只有被多个线程同时访问的可变数据才需要通过锁来保护。
以下几种变量不用加锁:
1、局部变量
2、用final修饰的全局变量不用加锁
3、单线程中的变量不用加锁