synchronized
synchronized是Java的一个关键字,它能够将代码块(方法)锁起来
只要在代码块(方法)添加关键字synchronized,即可以实现同步
synchronized是一种互斥锁
- 一次只能允许一个线程进入被锁住的代码块
synchronized是一种内置锁/监视器锁
- Java中每个对象都有一个内置锁(监视器,也可以理解成锁标记),而synchronized就是使用**对象的内置锁(监视器)**来将代码块(方法)锁定的!
同步代码块:
- monitorenter和monitorexit指令实现的
同步方法(在这看不出来需要看JVM底层实现)
- 方法修饰符上的ACC_SYNCHRONIZED实现。
synchronized底层是通过monitor对象,对象有自己的对象头,存储了很多信息,其中一个信息标示是被哪个线程持有。
synchronized一般我们用来修饰三种东西:
- 修饰普通方法(对象内置锁)
- 修饰代码块(某一对象内置锁)(什么是客户端锁)
- 修饰静态方法(类锁)
synchronized修饰静态方法获取的是类锁(类的字节码文件对象),synchronized修饰普通方法或代码块获取的是对象锁。
- 它俩是不冲突的,也就是说:获取了类锁的线程和获取了对象锁的线程是不冲突的!
内置锁的可重入性,锁的持有者是线程
Lock锁(是一个接口)
- Lock方式来获取锁支持中断、超时不获取、是非阻塞的
- 提高了语义化,哪里加锁,哪里解锁都得写出来
- Lock显式锁可以给我们带来很好的灵活性,但同时我们必须手动释放锁
- 支持Condition条件对象
- 允许多个读线程同时访问共享资源
对比
JDK1.6开始Synchronized锁就做了各种的优化:
- 优化操作:适应自旋锁,锁消除,锁粗化,轻量级锁,偏向锁。
- 详情可参考:https://blog.csdn.net/chenssy/article/details/54883355
到现在Lock锁和Synchronized锁的性能其实差别不是很大!而Synchronized锁用起来又特别简单。Lock锁还得顾忌到它的特性,要手动释放锁才行(如果忘了释放,这就是一个隐患)
所以说,我们绝大部分时候还是会使用Synchronized锁,用到了Lock锁提及的特性,带来的灵活性才会考虑使用Lock显式锁。
公平锁
公平锁理解起来非常简单:
- 线程将按照它们发出请求的顺序来获取锁
非公平锁就是:
- 线程发出请求的时可以**“插队”**获取锁
Lock和synchronize都是默认使用非公平锁的。如果不是必要的情况下,不要使用公平锁
- 公平锁会来带一些性能的消耗的