就我们之前提到的volatile时,重点就没有原子性这点有过讲解,而本篇要讲的Synchronized就与它相反,其中重要的一种就是原子性,其他特性还有可见性、有序性和可重入性。在接下来的内容中会分别进行阐释,找出一些使用Synchronized的要点,以助于我们更好的对它进行理解。
1.原子性
所谓原子性就是指一个操作或者多个操作,要么全部执行并且执行的过程不会被任何因素打断,要么就都不执行。
被synchronized修饰的类或对象的所有操作都是原子的,因为在执行操作之前必须先获得类或对象的锁,直到执行完才能释放,这中间的过程无法被中断(除了已经废弃的stop()方法),即保证了原子性。
注意:面试时经常会问比较synchronized和volatile,它们俩特性上最大的区别就在于原子性,volatile不具备原子性。
2.可见性
可见性:保证共享变量的修改能够及时可见,其实是通过Java内存模型中的 “对一个变量unlock操作之前,必须要同步到主内存中;如果对一个变量进行lock操作,则将会清空工作内存中此变量的值,在执行引擎使用此变量前,需要重新从主内存中load操作或assign操作初始化变量值”来保证的;
synchronized和volatile都具有可见性,其中synchronized对一个类或对象加锁时,一个线程如果要访问该类或对象必须先获得它的锁,而这个锁的状态对于其他任何线程都是可见的,并且在释放锁之前会将对变量的修改刷新到主存当中,保证资源变量的可见性,如果某个线程占用了该锁,其他线程就必须在锁池中等待锁的释放。
3.有序性
有效解决重排序问题,即 “一个unlock操作先行发生(happen-before)于后面对同一个锁的lock操作”;
synchronized和volatile都具有有序性,Java允许编译器和处理器对指令进行重排,但是指令重排并不会影响单线程的顺序,它影响的是多线程并发执行的顺序性。synchronized保证了每个时刻都只有一个线程访问同步代码块,也就确定了线程执行同步代码块是分先后顺序的,保证了有序性。
4.可重入性
synchronized和ReentrantLock都是可重入锁。当一个线程试图操作一个由其他线程持有的对象锁的临界资源时,将会处于阻塞状态,但当一个线程再次请求自己持有对象锁的临界资源时,这种情况属于重入锁。通俗一点讲就是说一个线程拥有了锁仍然还可以重复申请锁。public class SynchTest2 implements Runnable{
static SynchTest2 synchTest=new SynchTest2();
public void run() {
method1();
}
// 静态同步方法1
public synchronized void method1() {
System.out.println("开始执行同步方法1");
method2();
}
// 静态同步方法2
public synchronized void method2() {
System.out.println("开始执行同步方法2");
}
public static void main(String[] args) {
Thread t1=new Thread(synchTest);
t1.start();
while (t1.isAlive()) {
}
System.out.println("finished");
}
}
以上就是java中Synchronized特性的介绍,相信大家已经对Synchronized的使用有了进一步的了解,同时与之前的volatile方法有所区分,就原子性这一点需要大家在使用时注意。