Synchronized总结

2019-07-23  本文已影响0人  将代码写成诗

线程状态分析图

editByWpp.png

Synchronized的两种用法

 类锁的概念:
      Java类可能有多个对象,但只有1个Class对象
      - 本质:所以所谓的类锁,不过是Class对象的锁而已
      - 用法和效果:类锁只能在同一时刻被一个对象拥有
类锁的形式
    1.synchronized 放到方法上
    2.synchronized (*.class)代码块

多线程访问同步方法的7种情况

1.一把锁只能同时被一个线程获取,没有拿到锁的线程必须等待(对应1、5)
2.每个实例都对应有自己的一把锁,不同锁实例之间互不影响;例外:锁对象是*.class以及synchronized修饰的静态方法的时候,所有对象共用同一把类锁(对应2、3、4、6)
3.无论是方法正常执行完毕还是方法抛出异常,都会抛出异常(对应7)
注意:被synchronized修饰的方法a中调用另一个没有被synchronized修饰的方法b时,b可以被别的线程访问,此时方法b不受synchronized保护

性质

什么是可重入:指的是同一线程的外层函数获得锁之后,内层函数可以直接
再次获取锁
好处:避免死锁、提升封装性
粒度:线程而非调用(用三种情况来说明和pthread的区别)

粒度

不可中断

一旦这个锁被别人获得了,如果我还想获得,我只能选择等待或者阻塞,直到别的线程释放这个锁。如果别人永远不释放锁,那么我只能永远的等待下去。

相比之下,和Lock类比较,Lock可以拥有中断的能力,第一点,如果我觉得我等待时间太长了,我有权中断现在已经获得锁的线程的执行;第二点 如果我觉得我等的时间太长了 我可以选择退出等待状态。

原理

加锁和释放锁的原理

加锁和释放锁的原理:深入JVM看字节码
1、 概况--java虚拟机规范中有个monitor对象,该是对synchronized的实现的关键;monitor有个enter和exit
2、如何反编译呢?
javac xxx.java 编译 javap -verbose xxx.class 反编译(verbose 打印详细信息)
3.Monditorenter 和 Monditorexit指令

Monditorenter 
每个对象都与一个montior有关、每个monitor的lock锁在同一时间只能被一个线程获得;
当一个线程在尝试获得关于某个线程的所有权的时候,会出现一下三种情况之一:
1、当这个计数器为0时,表示这个锁处于闲置状态,这个线程会立刻获得这把锁,同时将计数器加一,
    表示获得这把锁的所有权,别的线程就不能再获得这把锁了;
2、(可重入性原则)当这个线程重入的时候,会导致该计数器加一;
3、当某线程尝试获取某把锁的时候,发现该对应的计数器计数器不为0,那么该线程就会进入阻塞状态,等待获得锁的线程释放锁
Monditorexit
每执行一次Monditorenter  该计数器就减一,若减一之后不为0,继续持有锁,为0则会释放锁,别的处于阻塞状态的线程就可以去抢这把锁了
可重入性原理

可见性的原理(内存模型)

image.png

缺陷

注意点:

建议:优先用java提供的类,其次用synchronized ,最后当需求有对锁的特殊需求,比如获得是否获得锁,给锁加释放时机的时候 再用Lock

思考点:

1、多个线程等待同一个synchronized锁的时候,JVM如何选择下一获取锁的的是哪一个线程呢?(内部锁调度机制)
2、synchronized使得同时只有一个线程可以执行,性能较差,有什么办法可以提升性能? 优化锁定范围,使用别的锁(比如读写锁)
3、想更灵活地控制锁的获取和释放(现在释放锁的时机都被规定死了)怎么办?
4、什么是锁的升级、降级?什么是JVM里的偏斜锁、轻量级锁、重量级锁?

总结:

一句话介绍synchronized

上一篇 下一篇

猜你喜欢

热点阅读