可重入锁即我调用加锁方法,该方法又调用了另一个加锁方法,相当于锁了两次(N次),但是不会出现死锁。
可重入锁:递归锁,外层函数获取锁之后,内层函数可以自动获取该锁,解决死锁问题
synchronized和reentrantLock是可重入锁
先说下sync:
必须是同一个线程,同一把锁才可以。
例子:
新建线程访问m1,m1打印后调用m2,两方法均锁定当前对象,并且均为同一线程,所以两方法可同时运行
——————————————————————————————————
其他例子1:如果不是同一个线程访问 m2阻塞
其他例子2:如加锁和未加锁同时运行,不会阻塞
其他例子3:锁的不是同一个对象,线程不会阻塞。
reentrantlock实现可重入锁
和sync一样,值得注意的是,sync是隐式锁,释放锁的过程无需我们设置,但reentrantlock是显式锁,必须手动解锁,否则会造成死锁!!!
例子1:同一线程调用同锁不同方法实现可重入锁
例子2:非同一线程会造成阻塞
例子3:如未释放锁,死锁
reentrantlock对synchronized的优化:
1.可以尝试申请锁
如果获取锁的时候是空闲的,那么线程访问,否则什么也不做
也可以指定时间,例多久之后我获取到锁,而后操作
2.lockInterruptibly()==》可以被打断的加锁
sync一旦wait,必须通过锁对象上的其他线程notify才可以
3.reentrantlock可以设置成公平锁,默认非公平锁
static Lock lock = new ReentrantLock(true);
后期会对底层实现原理进行整理