1.synchronized首先定义为一个重量锁(相对于偏向锁,轻量锁)java
2.使用场景:修饰普通方法;修饰静态方法;修饰代码块安全
3.修饰普通方法多线程
private int value; /* synchronized 修饰普通方法时,内置锁就是当前类的实例 */ public synchronized int getNext(){ return value++; }
首先说明下 value++这个操做是非原子性的(即包括从堆中取变量value,而后实现+1,最后写回堆中,这样多线程执行才会有安全性问题)this
而增长该关键字之后,进入这个方法时,都须要验证锁,即没有持有当前类实例的不容许进入,从而保证操做的原子性线程
4.修饰静态方法对象
/* 当synchronized关键字修饰静态方法时,内置锁是当前的Class字节码对象 solutionSaveProblemBySynchronized。class */ public static int getProvious(){ return value--; }
这种修饰时,锁是.class文件,即类的字节码文件,该文件具备惟一性,从而实现原子操做get
5.修饰代码块cmd
public int doSomeThing(){ /* 若是锁的是一个对象,则持有该对象的均可以进入,若是锁的是字节码文件 字节码文件具备惟一性 this指代当前对象 查看字节码文件指令: cmd 找到相应的。class文件,而后javap -verbose 字节码文件名 */ synchronized (this){ } synchronized (solutionSaveProblemBySynchronized.class){ } if(value>0){ return value; } else { return 0; } }
修饰代码块的时候,实际上是将要执行的代码体方式关键字的包裹中,能够说是放在一个方法中,synchronized()接收的是一个对象做为锁,去寻找对象的头部信息中获取锁信息,同步
关于对象头,能够参考其余做者博文,我也是去看了下其余做者的博文进行了解的io
续:1.synchronized属于可重入锁
2.valatile与synchronized的区别
1).都可实现变量可见性,从而实现同步
2).valatile对于非原子操做没法保证一致性,缘由是只是使变量可见,可是没有对非原子操做加锁
3).使用场景 一、状态标志量,二、double check