在Java中,synchronized关键字为防止资源冲突提供了支持,其作用域有二种:
  1. 实例范围。

对象实例范围内synchronized使用的两种形式:

实例范围同步方法

publicd class syncTest {
synchronized void aMethod() {
//需要同步使用的代码
}
}
synchronized aMethod(){}可以防止多个线程同时访问这个对象实例的synchronized方法(如果一个对象有多个synchronized方法,只要一个线程访问了其中的一个synchronized方法,其它线程不能同时访问这个对象中任何一个synchronized方法)。这时,不同的对象实例的synchronized方法是不相干扰的。也就是说,其它线程照样可以同时访问相同类的另一个对象实例中的synchronized方法。

实例范围同步区块

publicd class syncTest {
void aMethod() {
//无需同步使用的代码
synchronized(this) {
//需要同步使用的代码
}
}
}
除了方法前用synchronized关键字,synchronized关键字还可以用于方法中的某个区块中,表示只对这个区块的资源实行互斥访问。用法是: synchronized(this){/*区块*/},它的作用域是当前对象。
  1. 类范围

同样地,在类范围内使用synchronized也有同步方法和同步区块两种形式:

类范围同步方法

publicd class syncTest {
synchronized static void aMethod() {
//需要在类范围同步使用的代码
}
}
synchronized static aStaticMethod{}防止多个线程同时访问这个中的synchronized static 方法。它可以对类的所有对象实例起作用。也就是说在一个JVM中,同一时间最多有一个该类的静态同步方法在执行。这样的方法可以应用到多线程的同步中,实现各线程直接的数据共享和互动。

类范围同步区块

publicd class syncTest {
static void aMethod() {
//无需同步使用的代码
synchronized (syncTest.class) {
//需要在类范围同步使用的代码
}
}
}
类范围同步区块功能与类范围同步方法相同,只是对一个区块的代码的同步。用法是:在static 方法内加同步区块:sychronized (类名.class) { /*区块*/}
  1. 注意事项

synchronized关键字是不能继承的

synchronized关键字是不能继承的,也就是说,基类的方法synchronized f(){} 在继承类中并不自动是synchronized f(){},而是变成了f(){}。继承类需要你显式的指定它的某个方法为synchronized方法;

关于加锁

  • 无论synchronized关键字加在方法上还是对象上,它取得的锁都是对象,而不是把一段代码或函数当作锁――而且同步方法很可能还会被其他线程的对象访问;
  • 每个对象只有一个锁(lock)与之相关联;
  • 实现同步是要很大的系统开销作为代价的,甚至可能造成死锁,所以尽量避免无谓的同步控制。

对共享资源的同步访问更加安全的技巧

  • 定义private 的instance变量+它的 get方法,而不要定义public/protected的instance变量。如果将变量定义为public,对象在外界可以绕过同步方法的控制而直接取得它,并改动它;
如果instance变量是一个对象,如数组或ArrayList什么的,那上述方法仍然不安全,因为当外界对象通过get方法拿到这个instance对象的引用后,又将其指向另一个对象,那么这个private变量也就变了,这样是非常危险的。 这个时候就需要将get方法也加上synchronized同步,并且,只返回这个private对象的clone()。这样,调用端得到的就是对象副本的引用了。