6.锁
1.死锁
- 在线程间共享多个资源的时候,如果几个线程分别占有一部分资源不释放,并且大家都等待对方释放资源,就会造成死锁。(多个线程互相抱着对方需要的资源不放,形成僵持。)
- 某一个同步块同时拥有“两个以上对象的锁”时,就可能发生“死锁”的问题。
产生死锁的四个必要条件:
- 互斥条件:一个资源每次只能被一个进程使用。
- 请求和保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
- 不剥夺条件:进程已获得的资源,在未使用完之前,不能强行剥夺。
- 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。
避免死锁:
- 上面列出的4个条件,破坏其中的一个或多个,就能避免死锁。
- 不要在一个代码块当中,同时持有多个对象的锁(即不要锁套锁)
举例:
1 | /** |
2.Lock(锁)
- JDK5.0后,java提供了更强大的线程同步机制——通过显示定义同步锁对象来实现同步。同步锁使用Lock对象充当。
- java.util.concurrent.locks.Lock接口是控制多个线程对共享资源进行访问的工具。锁提供了对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源之前,先获得Lock对象。
ReentrantLock
类实现了Lock,它拥有与synchronized相同的并发性和内存语义。在实现线程安全的控制中, 比较常用的是ReentrantLock (可以显示加锁、释放锁)ReentrantLock
:可重用锁。
Lock大致格式:
1 | class A{ |
测试:
1.不安全写法:
1 | // 测试Lock锁 |
2.加lock锁 改成安全写法:
1 | // 测试Lock锁 |
3.synchronized 与 Lock对比
- Lock是显式锁(手动开启和关闭锁,别忘了关闭锁)。synchronized是隐式锁,出了作用域会自动释放锁。
- Lock只有代码块锁。synchronized有代码块锁和方法锁。
- 使用Lock锁,JVM将花费较少的时间来调度线程,性能更好。并且具有更好的扩展性(提供更多的子类)
- 优先使用顺序:
- Lock > 同步代码块(已进入了方法体,分配了相应资源)> 同步方法(在方法体之外)