Java多线程的可重入和不可重入

在 Java 中,可重入(Reentrant)和不可重入(Non-reentrant)这两个概念通常用于描述多线程环境下的同步方法或代码块的行为。

可重入(Reentrant)

可重入是指一个线程持有某个锁(Lock)时,可以再次请求同一个锁而不会被锁阻塞。换句话说,如果一个线程获取了某个对象的锁,它可以再次请求这个锁而不会导致死锁。

Java 中的 synchronized 关键字就是可重入的。当一个线程获取了某个对象的锁,它再次请求这个锁时,由于是同一个线程,所以会成功获取,而不会导致自己阻塞自己。这种特性使得 synchronized 可以用于递归方法或嵌套同步块。

不可重入(Non-reentrant)

不可重入是指当一个线程持有某个锁时,如果它再次尝试获取这个锁,它将被阻塞,直到锁被释放。这可能导致死锁,因为线程在等待自己释放锁,而永远不会释放。

不可重入锁的一个例子是 Java 中的 ReentrantLock(在默认情况下)。如果一个线程没有正确地释放锁,它再次尝试获取同一个锁时,就会发生死锁。

含义或意义

  • 可重入性:确保了线程在持有锁的情况下可以安全地调用其他需要相同锁的方法或代码块,这对于递归方法和内部调用非常重要。
  • 避免死锁:可重入锁可以避免由于锁的不当使用而导致的死锁问题。

区别

  1. 锁的请求:可重入锁允许线程在持有锁的情况下再次请求同一个锁,而不可重入锁不允许这样做。
  2. 死锁风险:不可重入锁在不当使用时可能导致死锁,而可重入锁则不会。
  3. 使用场景:可重入锁适用于需要递归调用或嵌套调用的场景,而不可重入锁可能需要更严格的锁管理策略。

示例

在 Java 多线程编程中,“可重入”和“不可重入”是指锁的特性,尤其是在使用 ReentrantLocksynchronized 关键字时。

可重入锁(Reentrant Lock)

可重入锁是指同一个线程可以多次获取同一个锁而不会导致死锁。每次获取锁时,锁的计数器会增加,释放锁时计数器会减少,直到计数器为零时,锁才会被真正释放。

示例代码
class ReentrantLockExample {
    private final Object lock = new Object();

    public void methodA() {
        synchronized (lock) {
            System.out.println("In methodA");
            methodB(); // 这里可以再进入 methodB
        }
    }

    public void methodB() {
        synchronized (lock) {
            System.out.println("In methodB");
        }
    }

    public static void main(String[] args) {
        ReentrantLockExample example = new ReentrantLockExample();
        example.methodA();
    }
}
解释
  • methodA 中,我们获取了 lock 的锁,然后调用 methodB
  • methodB 也尝试获取相同的 lock 锁。由于是可重入的,当前线程可以成功进入 methodB,不会产生死锁。
  • 输出将是:
    In methodA
    In methodB
    

不可重入锁(Non-Reentrant Lock)

不可重入锁是指同一个线程在获取锁后,如果再次请求相同的锁,则会导致死锁。这种锁通常不允许同一个线程多次获取。

示例代码
class NonReentrantLockExample {
    private final Object lock = new Object();

    public void methodA() {
        synchronized (lock) {
            System.out.println("In methodA");
            methodB(); // 试图进入 methodB
        }
    }

    public void methodB() {
        synchronized (lock) {
            System.out.println("In methodB");
        }
    }

    public static void main(String[] args) {
        NonReentrantLockExample example = new NonReentrantLockExample();
        example.methodA();
    }
}
解释
  • methodA 中,我们获取了 lock 的锁,然后调用 methodB
  • methodB 也尝试获取相同的 lock 锁。由于是不可重入的,当前线程会在这里阻塞,导致死锁,最终会抛出 java.lang.Exception
  • 你可能会看到如下输出:
    In methodA
    

可重入锁的作用

  1. 避免死锁:可重入锁允许一个线程多次获取同一把锁,避免了死锁的发生。
  2. 简化编码:在需要递归调用或者多个方法间互相调用的情况下,使用可重入锁可以简化代码的复杂性。

不可重入锁的作用

  1. 限制共享资源的访问:有时需要确保同一时间只有一个线程可以访问资源,避免数据的不一致性。
  2. 实现特定的同步需求:在一些特定的应用场景下,使用不可重入锁可以更好地控制线程的行为。

总结

  • 可重入锁:同一线程可以多次获取,避免死锁,适合复杂的调用关系。
  • 不可重入锁:同一线程不能多次获取,适合需要严格控制资源访问的场景。

根据实际需求选择合适的锁机制是多线程编程中重要的一部分。

  • 15
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值