随笔 - 465  文章 - 0  评论 - 7  阅读 - 27万

廖雪峰Java11多线程编程-2线程同步-3死锁

在多线程编程中,要执行synchronized块,必须首先获得指定对象的锁。

1.Java的线程锁是可重入的锁

    public void add(int m){
        synchronized (lock){
            this.value += m;
        }
    }

什么叫可重入的锁?
对同一个对象,同一个线程,可以多次获取它的锁,即同一把锁可以嵌套。
在add方法中,用synchronized获取了一个lock对象的锁,在synchronized代码块内部,调用another方法,可以再次获取到lock对象的锁。因为这是同一个线程对lock对象获取锁,所以它是一种可重入的锁。

    public void add(int m){
        synchronized (lock){
            this.value += m;
            another(m);
        }
    }
    public void addAnother(int m){
        synchronized (lock){
            this.another += m;
        }
    }
    //上述代码可以合并为
    public void add(int m){
        synchronized (lock){
            this.value += m;
            synchronized (lock){
                this.another += m;
            }
        }
    }

2. Java的线程还可以获取多个不同对象的锁

先用synchronized(lockA)获得lockA的锁,在用synchronized(lockB)获得lockB的锁。在释放锁的时候,会依次释放lockB的锁,再释放lockA的锁。

    public void add(int m){
        synchronized (lockA){//获取lockA的锁
            this.value += m;
            synchronized (LockB){//获取lockB的锁
                this.another += m;
            }//释放lockB的锁
        }//释放lockA的锁
    }

3.死锁

不同的线程获取多个不同对象的锁可能导致死锁
线程1和线程2分别执行。
线程1获取lockA的锁,线程2获得lockB的锁。
线程1执行语句,线程2执行语句。
线程1等待lockB的锁,线程2等待lockA的锁。这个时候,线程1和线程2会永远等待下去,谁也无法继续执行,就形成了死锁。

死锁形成的条件:

  • 两个线程各自持有不同的锁。线程1持有A锁,线程2持有B锁。
  • 两个线程各自试图获取对方的锁。线程1等待B锁,线程2等待A锁。
  • 双方无限等待下去,导致死锁

死锁发生后:

  • 没有任何机制能解除死锁
  • 只能强制结束JVM进程

所以,我们在编写多线程程序时,必须要避免出现死锁。
如何避免死锁:线程获取锁的顺序要一致

4.示例

4.1 死锁

class ShareObject{
    final Object lockA = new Object();
    final Object lockB = new Object();
    int accountA = 1000;
    int accountB = 2000;
    public void a2b(int balance){ //a2b: acountA减, accountB加
        synchronized (lockA){
            accountA -= balance;
            synchronized (lockB){
                accountB += balance;
            }
        }
    }
    public void b2a(int balance){ //b2a:acountA加, accountB减
        synchronized (lockB){
            accountB -= balance;
            synchronized (lockA){
                accountA += balance;
            }
        }
    }
}
class AThread extends Thread{ 
    public void run(){
        for(int i=0;i<Main.LOOP;i++){
            Main.shared.a2b(1);
            if(i%100==0){
                System.out.println(".");
            }
        }
    }
}
class BThread extends Thread{
    public void run(){
        for(int i=0;i<Main.LOOP;i++){
            Main.shared.b2a(1);
            if(i%100==0){
                System.out.println(".");
            }
        }
    }
}
public class Main{
    final static int LOOP = 1000;
    public static ShareObject shared = new ShareObject();
    public static void main(String[] args) throws Exception{
        Thread t1 = new AThread();
        Thread t2 = new BThread();
        t1.start();
        t2.start();
        t1.join();
        t2.join();
        System.out.println("end");
    }
}
### 4.2 避免死锁 修改b2a() ```#java public void b2a(int balance){ synchronized (lockA){ accountA += balance; synchronized (lockB){ accountB -= balance; } } } } ```

4总结

  • 死锁产生的条件:
    * 多线程各自持有不同的锁,并互相试图获取对方已持有的锁
  • 如何避免死锁?
    * 多线程获得锁的顺序要一致
posted on   singleSpace  阅读(240)  评论(0编辑  收藏  举报
编辑推荐:
· 深入理解 Mybatis 分库分表执行原理
· 如何打造一个高并发系统?
· .NET Core GC压缩(compact_phase)底层原理浅谈
· 现代计算机视觉入门之:什么是图片特征编码
· .NET 9 new features-C#13新的锁类型和语义
阅读排行:
· Spring AI + Ollama 实现 deepseek-r1 的API服务和调用
· 《HelloGitHub》第 106 期
· 数据库服务器 SQL Server 版本升级公告
· 深入理解Mybatis分库分表执行原理
· 使用 Dify + LLM 构建精确任务处理应用
< 2025年1月 >
29 30 31 1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31 1
2 3 4 5 6 7 8

点击右上角即可分享
微信分享提示