1.悲观锁和乐观锁
一个共享数据加了悲观锁,那线程每次想操作这个数据前都会假设其他线程也可能会操作这个数据,所以每次操作前都会上锁,这样其他线程想操作这个数据拿不到锁只能阻塞了。
synchronized
和 ReentrantLock是典型的悲观锁
共享数据加了乐观锁,一个线程操作数据的时候不会上锁,只会判断一下是否有其他线程修改了这个数据,所以乐观锁适合读多写少的场景,因为不用上锁、释放锁,省去了锁的开销,从而提升了吞吐量
乐观锁可以使用版本号机制和CAS算法来实现
2.独占锁和共享锁
独占锁
是指锁一次只能被一个线程所持有。如果一个线程对数据加上排他锁后,那么其他线程不能再对该数据加任何类型的锁。获得独占锁的线程即能读数据又能修改数据
synchronized是独占锁
共享锁
是指锁可被多个线程所持有。如果一个线程对数据加上共享锁后,那么其他线程只能对数据再加共享锁,不能加独占锁。获得共享锁的线程只能读数据,不能修改数据
3.互斥锁和读写锁
互斥锁是独占锁的一种实现,指某一资源同时只允许一个访问者对其进行访问,一次只能一个线程拥有互斥锁,其他线程只有阻塞等待
读写锁
是共享锁的一种实现,读写锁管理一组锁,一个是只读的锁,一个是写锁
读锁可以在没有写锁的时候被多个线程同时持有,而写锁是独占的
4.公平锁和非公平锁
公平锁
是指多个线程按照申请锁的顺序来获取锁
非公平锁
是指多个线程获取锁的顺序并不是按照申请锁的顺序,有可能后申请的线程比先申请的线程优先获取锁
synchronized 关键字是非公平锁,ReentrantLock默认也是非公平锁,但是ReentrantLock可以设置成公平锁:
/**
* 创建一个可重入锁,true 表示公平锁,false 表示非公平锁。默认非公平锁
*/
Lock lock = new ReentrantLock(true);
5.可重入锁
又称为递归锁,一个线程在外层方法获取了锁,进入内层方法会自动获取锁
ReentrantLock和Synchronized都是可重入锁
下面的代码中对方法A和方法B都加上了锁
如果一个线程调用了methodA,获取了A方法的锁,那就不用再特地去获取方法B的锁,自动获得了方法B的锁
如果是不可入锁, 这个线程不一定能抢到方法B的锁
public synchronized void mehtodA() throws Exception
{
mehtodB();
}
public synchronized void mehtodB() throws Exception
{
}
6.自旋锁
指线程在没有获得锁时不是被直接挂起,而是执行一个忙循环,这个忙循环就是所谓的自旋
总结:不挂起而是忙循环(自旋)
自旋锁的目的是为了减少线程被挂起的几率,因为线程的挂起和唤醒很耗资源
7.分段锁
将锁的粒度进一步细化,不是对整个数组加锁,而是仅对数组中的一部分进行加锁
CurrentHashMap 底层就用了分段锁
8.偏向锁
偏向于第一个访问锁的线程