synchronized关键字用于解决线程同步,其可以用于方法和代码段,锁定的都是对象,范围是实例对象或类对象。每个对象有且只有一个锁与之相关。
锁提供了两种主要特性:互斥(原子性)(mutual exclusion) 和可见性(visibility)。互斥即一次只允许一个线程持有某个特定的锁,因此可使用该特性实现对共享数据的协调访问协议,这样,一次就只有一个线程能够使用该共享数据。可见性要更加复杂一些,它必须确保释放锁之前对共享数据做出的更改对于随后获得该锁的另一个线程是可见的 —— 如果没有同步机制提供的这种可见性保证,线程看到的共享变量可能是修改前的值或不一致的值,这将引发许多严重问题。
synchronized有以下几种使用方式:
- 同步实例方法public synchronized void method():它锁定的是调用这个同步方法的对象,并不能同步对象所属类其他实例访问此方法;如果方法较大,不适合用同步方法,将严重影响效率。
- 同步静态方法public synchronized static void method():它锁定的是这个同步方法所在类对象,所有使用此类对象为锁的同步方法或代码段都受此同步方法影响,但1所使用的对象锁不受此同步方法影响。
- 同步代码段:
class TestSynchronized{
private byte[] lock = new byte[0]; // 特殊的实例变量,用此做锁,创建成本最低。
public void method(Object o)
{
synchronized (this){...} // 与1作用相同。
synchronized (o){...} // 锁定对象o,谁获得对象o的锁谁执行。
synchronized (lock ){...} // 没有一个明确对象来作为锁时,创建一个特殊实例变量做为锁。
synchronized (TestSynchronized.class){...} // 与2作用相同。
}
}
synchronized除了加锁外还同步内存:事实上,synchronized清除线程内存数据,从“主”内存区域中读入数据,在“ 主”内存区域同步整个线程的内存,对于变量的任何改变可以安全地写到“主”内存区域中,因此synchronized要消耗更多资源。
当编写 synchronized 块时,有几个简单的准则可以遵循,这些准则在避免死锁和性能危险的风险
方面大有帮助:
- 使代码块保持简短。Synchronized 块应该简短 — 在保证相关数据操作的完整性的同时,尽量简短。把不随线程变化的预处理和后处理移出 synchronized 块。
- 不要阻塞。不要在 synchronized 块或方法中调用可能引起阻塞的方法,如InputStream.read()。
- 在持有锁的时候,不要对其它对象调用方法。这听起来可能有些极端,但它消除了最常见的死锁源头。