不同的线程在访问共享数据时,会因为交织进行而导致线程干扰和内存一致性错误。
每个对象都有一个内部锁与其对应。如果一个线程需要排他一致性访问对象的字段,它首先要在访问之前获得该对象的内部锁。当访问完成时需要释放该内部锁。线程在获得该锁和释放该锁期间称为拥有该锁。一旦线程拥有内部锁,其他任何线程都不能再获得该锁,它们在获得该锁时会被阻塞。
创建互斥性操作的方法是互斥语句。在Java中,实现互斥语句的关键字叫synchronized(同步),互斥语句的语法格式如下:
synchronized(lock){
//critical code for accessing shared data.
//...
}
这里lock是提供内部所的对象。这个语句是互斥代码的一般写法。另外往往整个方法需要进行互斥,这时就有所谓互斥方法。互斥方法根据方法类型的不同分为实例互斥方法和静态互斥方法。实例互斥方法的例子如下:
public synchronized void addName(String name){
//Adding name to a shared list.
}
互斥实例方法实际获得的是当前实例对象的内部锁,前面的这个实例方法相当于下面写法的互斥语句:
public void addName(String name){
synchronized(this){
//Adding name to a shared list.
}
}
静态互斥方法的例子如下:
Public class ClassA{
public static synchronized void addName(String name){
//Adding name to a shared list.
}
}
静态互斥方法实际获得的是当前类Class对象的内部锁,前面这个静态方法相当于下面写法的互斥语句:
Public class ClassA{
public static void addName(String name){
synchronized(this){
//Adding name to a shared list.
}
}