《Java synchronized死锁剖析:可重入锁及哲学家就餐问题探究》

《Java synchronized死锁剖析:可重入锁及哲学家就餐问题探究》

文章内容

各位朋友,咱们来聊聊多线程里synchronized涉及的死锁相关内容。上一回咱们初步认识了锁,知道用synchronized修饰代码块能把非原子操作变成原子操作,那synchronized只能修饰代码块吗?当然不是啦,它还能修饰方法呢,实例方法和静态方法都可以哦。

目录

  1. synchronized修饰方法
  2. synchronized是可重入锁
  3. 死锁
    3.1. 死锁的引入
    3.2. 哲学家就餐问题

1. synchronized修饰方法

下面是一些用synchronized修饰方法和代码块的示例:

synchronized public void increase(){    // synchronized修饰实例方法
    count++;
}
public void increase1(){
    synchronized (this){   // 用this作为锁对象
        count++;
    }
}

synchronized public static void increase2(){
    count++;
}
public void increase3(){
    synchronized (Demo1.class){  // 通过反射拿到这个类对象
        count++;
    }
}

一个.java文件经过编译会生成.class字节码文件,运行时.class文件会被加载到JVM里,此时JVM加载到内存中的数据结构就是类对象。解释:当JVM加载一个.class文件时,会在内存中创建对应的类对象结构,这个类对象包含类的属性、名称、类型、权限、类方法、继承关系、实现的接口等信息。

2. synchronized是可重入锁

那可重入锁到底啥意思呢?就是一个线程针对同一个对象连续加锁两次不会出现死锁情况。咱们用代码来举个例子:

synchronized (locker){
    synchronized (locker){
        count++;
    }
}

按理说这段代码可能会卡住,但synchronized是可重入锁,允许同一个线程连续给一个对象加锁。对象头里有个计数器,线程给对象加锁一次,计数器就加1。解释对象头和计数器:线程给对象加锁时,会在对象头的Mark Word里保存线程信息,每个对象都有对象头(Object Header),里面包含锁状态等元数据。一个线程连续给对象加锁时,计数器也在对象头的Mark Word里,这能保证同一个线程多次获取同一个对象的锁。解锁时:一个对象被加锁三次,计数器值是3,解锁时不是直接全释放,而是每解一次锁计数器减1,直到减到0才是真正释放锁。

3. 死锁

3.1. 死锁的引入

刚才知道了synchronized是可重入锁,那啥时候会出现死锁呢?看下面这段代码:

public static void main(String[] args) {

    Object locker1 = new Object();
    Object locker2 = new Object();
    Thread t1 = new Thread(() ->{
        synchronized (locker1){
            try {
                Thread.sleep(1000);
                // 这里休眠很关键,不然线程可能一下子就拿到两个锁了
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            // 嵌套加锁
            synchronized (locker2){
                System.out.println("t1加锁成功");
            }
        }


    });

    Thread t2 = new Thread(() -> {
        synchronized(locker2){
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
            // 嵌套加锁
            synchronized(locker1){
                System.out.println("t2 加锁成功");
            }
        }

    });

    t1.start();
    t2.start();

}

运行这段代码,结果啥都没打印。通过jconsole查看线程状态,能看到两个线程都没成功获取第二把锁。这属于嵌套加锁情况,线程A拿到locker1锁后又想拿locker2锁,就可能出现死锁。要是并列加锁(线程先释放前面的锁再获取下一把锁)就不会死锁,嵌套变并列可以通过改变代码结构实现。

3.2. 哲学家就餐问题

看下面这张图表示的哲学家就餐问题:

《Java synchronized死锁剖析:可重入锁及哲学家就餐问题探究》

这就导致了死锁问题。死锁形成通常有四个必要条件:
1. 互斥使用:锁的基本特性,一个线程拥有锁A,其他线程想获取就得阻塞等待。
2. 不可抢占:和条件一类似,其他线程只能等拥有锁的线程释放,不能强行抢占。
3. 保持请求:线程想获取多把锁(嵌套,想获取锁B但不想释放自己的锁),其实合理获取多把锁没问题,但会形成环导致死锁。
4. 循环等待:条件三导致的,等待关系形成环。

解决哲学家就餐问题可以通过约定加锁顺序,给锁编号,先加小锁再加大锁。比如规定每个哲学家先拿数字小的筷子,像哲学家B先拿筷子,最后哲学家A面前只剩数字大的筷子5,拿不了,而哲学家E就能吃到面条,这样就解决了死锁问题。

《Java synchronized死锁剖析:可重入锁及哲学家就餐问题探究》

以上就是synchronized死锁问题的全部内容啦,死锁会让程序陷入死循环,只要知道死锁成因就能找到解决办法啦~欲知后续如何,且听下回分解~

能看到这里说明您对文章有一定认可,有问题欢迎大佬指出,欢迎评论区留言修正哦~

文章整理自互联网,只做测试使用。发布者:Lomu,转转请注明出处:https://www.it1024doc.com/13580.html

(0)
LomuLomu
上一篇 2025 年 9 月 17 日
下一篇 2025 年 9 月 17 日

相关推荐

  • IDEA激活失败怎么办?附激活工具&使用技巧

    本教程适用于IDEA、PyCharm、DataGrip、Goland等,支持Jetbrains全家桶! 废话不多说,先上最新 IDEA 版本破解成功的截图,如下,可以看到已经成功破解到 2099 年辣,舒服! 接下来,我就将通过图文的方式, 来详细讲解如何激活 IDEA至 2099 年。 当然这个激活方法,同样适用于之前的旧版本! 不管你是什么操作系统,什么…

    IDEA破解教程 2025 年 9 月 24 日
    6600
  • [JAVA] 第十一章:深入浅出解析”继承”核心概念(新手入门指南)

    目录导航1. 类继承机制1.1 继承原理剖析1.2 继承语法规范1.3 访问父类成员 1.3.1 子类访问父类字段①子父类字段无冲突②子父类字段同名super关键字的妙用 1.3.2 子类调用父类方法①方法名不同②方法名相同1.4 子类构造器特性🔍构造器特点精要1.5 super与this对比✔共同特征✖差异分析1.6 初始化流程详解1.7 protecte…

    2025 年 5 月 12 日
    14600
  • Android Studio更改项目使用的JDK

    一、对配置流程的不满 在安卓开发领域,配置JDK和Gradle本应是一项简单的任务。通常情况下,开发者只需在Android Studio的File菜单下选择Project Structure,便能轻松完成设置,这一流程可谓便捷至极。 以下是原先修改JDK的界面: 然而,众所周知,Android Studio的性能问题一直为人所诟病,加之其对用户需求的忽视,使…

    2024 年 12 月 28 日
    77300
  • 【java-数据结构篇】揭秘 Java LinkedList:链表数据结构的 Java 实现原理与核心概念

    我的个人主页我的专栏:Java-数据结构 ,希望能帮助到大家!!!点赞❤ 收藏❤ 目录 1. Java LinkedList 基础 1.1 LinkedList 简介1.2 LinkedList 的实现原理1.3 LinkedList 与 ArrayList 的区别 2. 链表基础 2.1 链表的定义与种类2.2 单链表与双链表的区别2.3 循环链表与普通链…

    2024 年 12 月 28 日
    36600
  • 2025年最新IDEA激活码分享 | IDEA永久破解教程及注册码获取指南

    全面支持Jetbrains全家桶的破解方案 本教程适用于IntelliJ IDEA、PyCharm、DataGrip、Golang等Jetbrains全系列开发工具,让您轻松实现永久激活! 先来看看最新IDEA版本成功破解的截图展示,可以看到有效期已延长至2099年,完美解决激活问题! 下面将详细介绍如何通过简单几步操作完成IDEA的永久激活。这个方法不仅适…

    IDEA破解教程 2025 年 7 月 17 日
    40400

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信