synchronized关键字是java并发中非常常用,其作用为保证被修饰的方法,对象或者代码块在同一时间只会被一个线程持有,也就是实现了互斥。

应用举例

修饰实例方法

public synchronized void method(){
}

synchronized修饰实例方法,锁住的是当前实例对象,并不是锁住了该类,因此该类的静态方法依然是可以被其他线程持有的。

修饰静态方法

public static synchronized void method(){
}

synchronized修饰类的静态方法,锁住的是当前类对象。该类对象的所有实例对象都会被锁住。

修饰实例对象

synchronized(this){
}

synchronized修饰一个类的实例对象,锁住当前实例对象。

修饰class对象

synchronized(Demo.class){
}

synchronized修饰一个类对象,锁住该类对象以及所有此类的实例对象。

修饰自定义对象

String lock = " ";
synchronized(lock){
}

可以自定义一个任意实例对象,实现对代码块的同步。该做法常用的一个场景是,对数组类型的实例进行操作,通过修饰该数组实例实现同步操作。

番外之volatile关键字

volatile也是在并发中常用的关键字,用来修饰变量。

public volatile int a;

被修饰的变量在一个线程中被修改后,其余线程会立刻知道该变量已经得到了修改,并自动去共享内存区将该变量拷贝到自己的工作内存区,也就是实现了多线程之间的数据同步。

和synchronized相比,volatile不保证原子性,因此该关键字的主要目的是实现变量在多个线程中的可见性,synchronized则实现了多个线程中的资源访问同步性。

volatile的常用情景为:多个线程共享一个变量,共同对其修改;或者是将该变量作为触发器,实现线程中的轻量级同步。

此外,volatile可以避免代码优化,防止重排序问题出现,可以用来解决多线程安全的延迟初始化问题,在这方面和c++中的volatile功能类似。因此java的volatile可以视作c++的volatile的升级版。