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