就我们之前提到的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方法有所区分,就原子性这一点需要大家在使用时注意。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)