可重入锁synchronized 和 ReentrantLock

朴灿烈づ我的快乐病毒、 2022-12-29 14:23 214阅读 0赞

JVM的 synchronized 是可重入锁,即可以多次获得相同的锁(就是说某个线程已经获得某个锁,可以再次获取锁而不会出现死锁)

  1. new Thread(new Runnable() {
  2. @Override
  3. public void run() {
  4. synchronized (this) {
  5. System.out.println("第1次获取锁,这个锁是:" + this);
  6. int index = 1;
  7. while (true) {
  8. synchronized (this) {
  9. System.out.println("第" + (++index) + "次获取锁,这个锁是:" + this);
  10. }
  11. if (index == 10) {
  12. break;
  13. }
  14. }
  15. }
  16. }
  17. }).start();

ReentrantLock 可重入锁,但是需要手动释放

使用ReentrantLock的注意点

ReentrantLock 和 synchronized 不一样,需要手动释放锁,所以使用 ReentrantLock的时候一定要手动释放锁,并且加锁次数和释放次数要一样

以下代码演示,加锁和释放次数不一样导致的死锁

  1. package com.test.reen;
  2. import java.util.Random;
  3. import java.util.concurrent.locks.ReentrantLock;
  4. public class WhatReentrant3 {
  5. public static void main(String[] args) {
  6. ReentrantLock lock = new ReentrantLock();
  7. new Thread(new Runnable() {
  8. @Override
  9. public void run() {
  10. try {
  11. lock.lock();
  12. System.out.println("第1次获取锁,这个锁是:" + lock);
  13. int index = 1;
  14. while (true) {
  15. try {
  16. lock.lock();
  17. System.out.println("第" + (++index) + "次获取锁,这个锁是:" + lock);
  18. try {
  19. Thread.sleep(new Random().nextInt(200));
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. }
  23. if (index == 10) {
  24. break;
  25. }
  26. } finally {
  27. // lock.unlock();// 这里故意注释,实现加锁次数和释放次数不一样
  28. }
  29. }
  30. } finally {
  31. lock.unlock();
  32. }
  33. }
  34. }).start();
  35. new Thread(new Runnable() {
  36. @Override
  37. public void run() {
  38. try {
  39. lock.lock();
  40. for (int i = 0; i < 20; i++) {
  41. System.out.println("threadName:" + Thread.currentThread().getName());
  42. try {
  43. Thread.sleep(new Random().nextInt(200));
  44. } catch (InterruptedException e) {
  45. e.printStackTrace();
  46. }
  47. }
  48. } finally {
  49. lock.unlock();
  50. }
  51. }
  52. }).start();
  53. }
  54. }

由于加锁次数和释放次数不一样,第二个线程始终无法获取到锁,导致一直在等待。

稍微改一下,在外层的finally里头释放9次,让加锁和释放次数一样,就没问题了

  1. try {
  2. lock.lock();
  3. System.out.println("第1次获取锁,这个锁是:" + lock);
  4. int index = 1;
  5. while (true) {
  6. try {
  7. lock.lock();
  8. System.out.println("第" + (++index) + "次获取锁,这个锁是:" + lock);
  9. ... 代码省略节省篇幅...
  10. } finally {
  11. // lock.unlock();// 这里故意注释,实现加锁次数和释放次数不一样
  12. }
  13. }
  14. } finally {
  15. lock.unlock();
  16. // 在外层的finally里头释放9次,让加锁和释放次数一样,就没问题了
  17. for (int i = 0; i < 9; i++) {
  18. lock.unlock();
  19. }
  20. }

synchronized (this)和synchronized 修饰方法 Object o =new Object() synchronized (o) 功能都是等价的

synchronized 是对资源(家)上一把锁(锁的是对象,并不是代码),至于锁是什么,任意。谁争抢到那把锁,谁就有权利执行锁下面的代码(进家里)(家里的锁可以是任意的锁,只要把家锁好就成)

synchronized 既保证了原子性,又保证了可见性,因为线程持有该锁的时候,其它线程并不能被抢到

发表评论

表情:
评论列表 (有 0 条评论,214人围观)

还没有评论,来说两句吧...

相关阅读