Spring 依赖注入和自动装配

news2024/11/27 0:31:01

DI(依赖注入)

DI:Dependency Injection

共有三种方式

构造器注入

在前面IOC容器创建对象的方式中已经提到,无参构造器和有参构造器都可以。

Set方式注入(重点)

  • 依赖注入:本质是Set注入

    • 依赖:

      bean对象的创建依赖于容器

    • 注入:

      bean对象中的所有属性由容器来注入

环境搭建

  1. 复杂类型

    结构:

    Address实体类:

    package xyz.luck1y.pojo;
    
    public class Address {
        private String address;
    
        public String getAddress() {
            return address;
        }
    
        public void setAddress(String address) {
            this.address = address;
        }
    
        @Override
        public String toString() {
            return "Address{" +
                    "address='" + address + '\'' +
                    '}';
        }
    }
    
  2. 真实测试环境

    Student实体类:

    package xyz.luck1y.pojo;
    
    import java.util.*;
    
    public class Student {
        private String name;
        private Address address;
        private String[] books;
        private List<String> hobbys;
        private Map<String, String> card;
        private Set<String> games;
        private String wife;
        private Properties info;
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public Address getAddress() {
            return address;
        }
    
        public void setAddress(Address address) {
            this.address = address;
        }
    
        public String[] getBooks() {
            return books;
        }
    
        public void setBooks(String[] books) {
            this.books = books;
        }
    
        public List<String> getHobbys() {
            return hobbys;
        }
    
        public void setHobbys(List<String> hobbys) {
            this.hobbys = hobbys;
        }
    
        public Map<String, String> getCard() {
            return card;
        }
    
        public void setCard(Map<String, String> card) {
            this.card = card;
        }
    
        public Set<String> getGames() {
            return games;
        }
    
        public void setGames(Set<String> games) {
            this.games = games;
        }
    
        public String getWife() {
            return wife;
        }
    
        public void setWife(String wife) {
            this.wife = wife;
        }
    
        public Properties getInfo() {
            return info;
        }
    
        public void setInfo(Properties info) {
            this.info = info;
        }
    
        @Override
        public String toString() {
            return "Student{" +
                    "name='" + name + '\'' +
                    ", address=" + address +
                    ", books=" + Arrays.toString(books) +
                    ", hobbys=" + hobbys +
                    ", card=" + card +
                    ", games=" + games +
                    ", wife='" + wife + '\'' +
                    ", info=" + info +
                    '}';
        }
    }
    
  3. xml配置:

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans.xsd">
    
        <bean id="student" class="xyz.luck1y.pojo.Student">
            <property name="name" value="刘子"/>
        </bean>
    </beans>
    
  4. 测试环境:

    import org.springframework.context.support.ClassPathXmlApplicationContext;
    import xyz.luck1y.pojo.Student;
    
    public class MyTest {
        public static void main(String[] args) {
            ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
            Student student = (Student) context.getBean("student");
            System.out.println(student.getName());
        }
    }
    

开始测试

  1. xml配置:

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans.xsd">
    
        <bean id="address" class="xyz.luck1y.pojo.Address">
            <property name="address" value="南京"/>
        </bean>
    
        <bean id="student" class="xyz.luck1y.pojo.Student">
            <!--第一种,普通值注入,value-->
            <property name="name" value="刘子"/>
            <!--第二种,Bean注入-->
            <property name="address" ref="address"/>
            <!--第三种,数组注入-->
            <property name="books">
                <array>
                    <value>《红楼梦》</value>
                    <value>《水浒传》</value>
                    <value>《三国演义》</value>
                    <value>《西游记》</value>
                </array>
            </property>
            <!--第四种,list注入-->
            <property name="hobbys">
                <list>
                    <value>听歌</value>
                    <value>健身</value>
                    <value>敲代码</value>
                </list>
            </property>
            <!--第五种,map注入-->
            <property name="card">
                <map>
                    <entry key="身份证" value="222303220222122222"/>
                    <entry key="学生证" value="202241803119"/>
                </map>
            </property>
            <!--第六种,set集合注入-->
            <property name="games">
                <set>
                    <value>LOL</value>
                    <value>永劫无间</value>
                    <value>瓦罗兰特</value>
                </set>
            </property>
            <!--第七种,Null值注入-->
            <property name="wife">
                <null/>
            </property>
            <!--第八种,properties-->
            <property name="info">
                <props>
                    <prop key="driver">jdbcDriver</prop>
                    <prop key="url">https://www.baidu.com</prop>
                    <prop key="username">root</prop>
                    <prop key="password">123456</prop>
                </props>
            </property>
        </bean>
    </beans>
    
  2. 测试

    import org.springframework.context.support.ClassPathXmlApplicationContext;
    import xyz.luck1y.pojo.Student;
    
    public class MyTest {
        public static void main(String[] args) {
            ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
            Student student = (Student) context.getBean("student");
            System.out.println(student);
            /*
            Student{
            name='刘子',
            address=Address{address='南京'},
            books=[《红楼梦》, 《水浒传》, 《三国演义》, 《西游记》],
            hobbys=[听歌, 健身, 敲代码],
            card={身份证=222303220222122222, 学生证=202241803119},
            games=[LOL, 永劫无间, 瓦罗兰特],
            wife='null',
            info={password=123456,
            driver=jdbcDriver,
            url=https://www.baidu.com,
            username=root}
            }
             */
        }
    }
    

拓展方式注入

P命名空间和C命名空间在配置中都需要导入约束。一个无参,一个有参。其实p就是property,c就是constructor-arg

xmlns:p="http://www.springframework.org/schema/p"
xmlns:c="http://www.springframework.org/schema/c"

P命名空间

实体类:

package xyz.luck1y.pojo;

public class User {
    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

配置xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:p="http://www.springframework.org/schema/p"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--P 命名空间注入:可以直接注入属性的值-->
    <bean id="user" class="xyz.luck1y.pojo.User" p:age="21" p:name="刘子">        
    </bean>
</beans>

测试:

@Test
public void test2(){
    ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("userbeans.xml");
    // 如果显式地声明类型则不用强转
    Object user = context.getBean("user", User.class);
    System.out.println(user);
}

C命名空间

实体类需要增加有参构造:

package xyz.luck1y.pojo;

public class User {
    private String name;
    private int age;

    public User() {
    }

    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

xml配置:

<!--C 命名空间注入:通过构造器注入:construct-args-->
<bean id="user2" class="xyz.luck1y.pojo.User" c:age="22" c:name="刘同学">

</bean>

测试:

@Test
public void test2(){
    ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("userbeans.xml");
    // 如果显式地声明类型则不用强转
    Object user = context.getBean("user2", User.class);
    System.out.println(user);
}

这两种拓展命名方式需要导入相应的约束~

Bean的作用域

  1. 单例模式(Spring默认机制)

    每一次getBean得到的对象都是同一个,hashcode都是相同的

    <bean id="user2" class="xyz.luck1y.pojo.User" c:age="22" c:name="刘同学" scope="singleton">
    
    </bean>
    
  2. 原型模式

    每一次getBean得到的对象不一样,hashCode不同

    <bean id="user2" class="xyz.luck1y.pojo.User" c:age="22" c:name="刘同学" scope="prototype">
    
    </bean>
    
  3. 其余的request、session、application、websocket只会在web开发中使用到。

Bean的自动装配

  • 自动装配是Spring满足Bean依赖的一种方式
  • Spring会在上下文中自动寻找并自动给Bean装配属性

在Spring中有三种装配方式:

  1. 在xml中显式地配置
  2. 在Java中显式地配置
  3. 隐式地自动装配bean(重点)

测试

  1. 一个人有两只宠物:一只猫,一只狗

    实体类:

    People

    package xyz.luck1y.pojo;
    
    public class People {
        private Cat cat;
        private Dog dog;
        private String name;
    
        public Cat getCat() {
            return cat;
        }
    
        public void setCat(Cat cat) {
            this.cat = cat;
        }
    
        public Dog getDog() {
            return dog;
        }
    
        public void setDog(Dog dog) {
            this.dog = dog;
        }
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        @Override
        public String toString() {
            return "People{" +
                    "cat=" + cat +
                    ", dog=" + dog +
                    ", name='" + name + '\'' +
                    '}';
        }
    }
    

    Dog

    package xyz.luck1y.pojo;
    
    public class Dog {
        public void shout(){
            System.out.println("汪汪汪~");
        }
    }
    

    Cat

    package xyz.luck1y.pojo;
    
    public class Cat {
        public void shout(){
            System.out.println("喵喵喵~");
        }
    }
    
  2. 配置bean.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans.xsd">
        <bean id="cat" class="xyz.luck1y.pojo.Cat"/>
        <bean id="dog" class="xyz.luck1y.pojo.Dog"/>
    
        <bean id="people" class="xyz.luck1y.pojo.People">
            <property name="name" value="刘子"/>
            <!--ref(引用的)注入的是对象;value注入的是值-->
            <property name="cat" ref="cat"/>
            <property name="dog" ref="dog"/>
        </bean>
    </beans>
    
  3. 测试

    import org.junit.jupiter.api.Test;
    import org.springframework.context.support.ClassPathXmlApplicationContext;
    import xyz.luck1y.pojo.People;
    
    public class MyTest {
        @Test
        public void test(){
            ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
            People people = context.getBean("people", People.class);
            people.getDog().shout();
            people.getCat().shout();
        }
    }
    
  4. 结果:

ByName和ByType自动装配

ByName

byName:会自动在容器上下文查找,和自己属性set方法后面的值对应的beanID

比如cat属性的set方法为setCat,就会自动寻找id为cat的bean进行ref引用,Spring会将setCat方法的Cat自动转换为小写cat,byName只能取到id为小写的,不能取到id为大写的,并且需要保证id唯一。

<bean id="people" class="xyz.luck1y.pojo.People" autowire="byName">
    <property name="name" value="刘子"/>
</bean>

ByType

byType:会自动在容器上下文查找,和自己属性set方法参数的类型相同的beanID

但是如果有两个Cat属性的Bean 就会失败报错,必须保证类型全局唯一,甚至可以省略id进行装配

<bean id="people2" class="xyz.luck1y.pojo.People" autowire="byType">
    <property name="name" value="刘子"/>
</bean>

小结:

  • ByName的时候,需要保证所有bean的id唯一,并且这个bean需要和自动注入的属性的set方法的值一致。
  • ByType的时候,需要保证所有bean的class唯一,并且这个bean需要和自动注入的属性的类型一致。

使用注解实现自动装配

JDK从1.5开始支持注解,Spring从2.5开始支持注解。

使用注解:

  1. 在配置文件中导入约束:

    xmlns:context="http://www.springframework.org/schema/context"
    
  2. 配置开启注解的支持:

    <!--一定记得写这句,开启注解支持-->
    <context:annotation-config/>
    
    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xmlns:context="http://www.springframework.org/schema/context"
        xsi:schemaLocation="http://www.springframework.org/schema/beans
            https://www.springframework.org/schema/beans/spring-beans.xsd
            http://www.springframework.org/schema/context
            https://www.springframework.org/schema/context/spring-context.xsd">
    	
        <!--开启注解-->
        <context:annotation-config/>
    
    </beans>
    

@Autowired

在属性上使用也可以在set方法上使用

@Autowired
private Cat cat;
@Autowired
private Dog dog;

使用@Autowired注解后,我们甚至可以不用编写set方法,前提是要自动装配的属性在IOC(Spring)容器中存在,并且符合ByName

其实这个注解的实现默认就是ByType,当ByType查找不到或当ByType的满足个数大于1的时候就会使用ByName

科普:

@Nullable 属性标记了这个注解,说明这个属性可以为null

进入@Autowired可以看到是有一个属性的:

public @interface Autowired {
    boolean required() default true;
}

如果显示的定义了required属性为false,说明这个属性可以为null(默认不能为空),可以不用在bean中装配。一般来说包装类默认可以是null,不需要声明,但是一些基本属性想要默认为null的时候,可以这样声明。

@Autowired(required = false)
private Cat cat;

如果@Autowired自动装配的环境比较复杂,自动装配无法通过@Autowired一个注解来完成,我们可以通过组合@Autowired@Qualifier,指定具体的一个bean:如果同时有两个Dog类的bean,一个id为dog111,一个为dog222,这个时候byType查不到了,因为有两个,所以就去ByName,但是发现两个bean的id也跟setDog方法的“dog”不对应,这时候就不知道要用哪一个,会报错!

然后我们就可以通过@Qualifier指定ByName使用。

@Autowired
@Qualifier(value = "dog222")
private Dog dog;

拓展:

除了使用Spring的自动装配注解外,Java的jdk也提供了一个默认的装配注解:@Resource,但是jdk11取消了这个注解,因此存在版本兼容问题。这个注解默认是按照ByName进行匹配,匹配不到用ByType进行匹配。@Resource(name=“cat2”),同样的,这个注解也可以指定匹配一个bean。

小结

@Resource和@Autowired的区别:

  • 都是用来自动装配的,都可以放在属性或者set方法之前

  • @Resource:

    • 默认使用ByName匹配,**有同名直接报错!**通过Name匹配不到,再使用ByType进行匹配。
    • 可以指定使用name或type进行匹配给出的bean:@Resource(name=“”)或@Resource(Type=“”)
    • 如果指定name,就会去查找指定的值,如果查找不到,会直接报错,会自动屏蔽掉另外一种查询方式!如果找到了,会判断type,只有type也匹配才会成功。
  • @Autowired:

    • 默认使用ByType匹配,匹配不到直接报错!匹配数大于1或匹配到多个,再使用ByName进行匹配。

    • 可以指定属性是否为空:@Autowired(required=false),false:可以空,true:不能为空

    • 可以结合@Qualifier通过byName匹配给出的bean:

      @Autowired
      @Qualifier(value="")
      
    • 如果同时配置了@Qualifier(value=“”),就会去寻找配置的这个值,这时候找不到就直接报错,会自动屏蔽ByType匹配,如果找到了,会判断type,只有type也匹配才会成功。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/877643.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Linux】高级IO

目录 IO的基本概念 钓鱼五人组 五种IO模型 高级IO重要概念 同步通信 VS 异步通信 阻塞 VS 非阻塞 其他高级IO 阻塞IO 非阻塞IO IO的基本概念 什么是IO&#xff1f; I/O&#xff08;input/output&#xff09;也就是输入和输出&#xff0c;在著名的冯诺依曼体系结构当中…

LinuxC编程——进程间通信(二)(信号、共享内存)

目录 一、信号1.1 概念1.2 信号的响应方式⭐⭐⭐1.3 几种常见的信号1.4 函数练习 二、共享内存2.1 共享内存的特点2.2 共享内存创建步骤⭐⭐2.3 共享内存创建所需函数 信号主要用来通知进程异步事件的发生。最初信号设计的目的是为了处理错误&#xff0c;它们也用来作为最基本的…

【EI/SCOPUS检索】第二届能源与动力工程国际学术会议(EPE 2023)

第二届能源与动力工程国际学术会议&#xff08;EPE 2023&#xff09; 2023 2nd International Conference on Energy and Power Engineering 能源是人类社会发展的重要推动力量。如何安全、清洁、高效地存储、转化和利用能源&#xff0c;实现人类可持续发展&#xff0c;一直…

比例方向阀控制多功能放大器

适用于控制无电位置反馈的三位四通比例方向阀&#xff0c;两路独立工作的比例放大器&#xff0c;可组合成并联工作方式&#xff0c;0到10V输入接口&#xff0c;可切换为0(4)到20mA输入&#xff0c;工作电压24VDC&#xff0c;允许工作温度范围0~45℃&#xff0c;放大器只有在断电…

spring security实践-全套代码

贴一套完整代码 电脑文件都被加密了&#xff0c;无法上传git&#xff0c;留一套在此&#xff0c;日后方便。 整个学习过程参考的spring security 1. 项目目录结构 2.初始化数据库 CREATE TABLE sys_user (id BIGINT(20) NOT NULL AUTO_INCREMENT COMMENT 主键,user_name VAR…

你确定你的 REST API 真的符合 REST 规范?

RESTful API 的存在是 web 开发历史上的一个里程碑。在本文中&#xff0c;我将和你探讨几种节省 REST API 开发时间的方法&#xff0c;并给出相关的 Node.js 示例。 什么是 RESTful API 首先&#xff0c;想问一个问题&#xff0c;你的项目里真的有真正的 RESTful API 吗&…

【构造】CF1734 E

Problem - 1734E - Codeforces 题意&#xff1a; 思路&#xff1a; Code&#xff1a; #include <bits/stdc.h>#define int long longusing i64 long long;constexpr int N 1e3 10; constexpr int mod 1e9 7;int n; int a[N], ans[N][N];void solve() {std::cin &g…

动力节点零基础Java学习教程,从入门到进阶,轻松掌握Java开发技能

Java是一门应用非常广泛的编程语言&#xff0c;对于零基础自学Java来说&#xff0c;Java的学习过程可能会有一些困难&#xff0c;但只要掌握了相关的基础知识和技能&#xff0c;不断地实践和总结&#xff0c;就能真正掌握Java编程。 动力节点老杜的Java经典之作&#xff0c;哔站…

Jenkins构建自由风格项目发布jar到服务器

前面的文章有介绍 docker安装jenkins 和 dockerjenkins发布spring项目&#xff1b;这里就不做过多的介绍&#xff0c;直接说明构建步骤。 1、选择构建一个自由风格的项目 2、 选择丢弃旧的构建 3、配置Git信息 4、构建触发器 和 构建环境可以直接跳过 5、直接来到Build Step…

技术广度必备——高并发设计之分布式锁的实现方式

文章目录 问题背景前言实现基于MySQL实现唯一索引乐观锁悲观锁 基于Redis基于Zookeeper原理使用Curator框架实现ZK分布式锁缺点 问题背景 研究有哪几种方案可以实现分布式锁&#xff0c;技术选型的场景下能用到。 前言 本文参考过的文章有分布式锁的几种实现方式方式大致分为3种…

蓝帽杯 取证2022

网站取证 网站取证_1 下载附件 并解压 得到了一个文件以及一个压缩包 解压压缩包 用火绒查病毒 发现后门 打开文件路径之后 发现了一句话木马 解出flag 网站取证_2 让找数据库链接的明文密码 打开www文件找找 查看数据库配置文件/application/database.php&#xff08;CodeI…

Cuda和cuDNN安装

Cuda和cuDNN安装 1Cuda下载与安装1.1查看适合cuda的版本1.2下载cuda toolkit1.3cuda安装步骤1.4配置环境变量1.5验证 2cuDNN下载与安装2.1cuDNN下载2.2cuDNN配置2.3配置环境变量2.4验证 3安装PyTorch-GPU3.1打开Anaconda Prompt![在这里插入图片描述](https://img-blog.csdnimg…

ARM--day2(cpsr、spsr、数据搬移指令、移位操作指令、位运算操作指令、算数运算指令、比较指令、跳转指令)

.text .global _gcd _gcd:mov r0,#9mov r1,#15b loop loop:cmp r0,r1beq stopsubhi r0,r1bhi loopsubcc r1,r0bcc loopstop:b stop.end用for循环实现1~100之间和5050 .text .global _gcd _gcd:mov r0,#0x0mov r1,#0x1mov r2,#0x64b loop loop:cmp r1,r2bhi stopadd r0,r0,r1ad…

卷王特斯拉又全网降价了,卷死车企们

哈喽,大家好,今天媒介盒子小编又来跟大家分享软文推广的干货知识了,本篇分享的主要内容是&#xff1a;特斯无孔不入的营销手段。 1、特斯拉Model Y降价 车企要打架 自2023 年 8 月 14 日起&#xff0c;Model Y 长续航版起售价从 31.39 万元调整为 29.99 万元&#xff0c;Mode…

React+Typescript清理项目环境

上文 创建一个 ReactTypescript 项目 我们创建出了一个 React配合Ts开发的项目环境 那么 本文 我们先将环境清理感觉 方便后续开发 我们先来聊一下React的一个目录结构 跟我们之前开发的React项目还是有一些区别 public 主要是存放一些静态资源文件 例如 html 图片 icon之类的 …

走进 Linux

一、开关机 开机&#xff1a; 开机会启动许多程序。他们在windows叫做“服务”(service),在Linux就叫做“守护进程”&#xff08;daemon&#xff09;开机成功后&#xff0c;它会显示一个文本登录界面&#xff0c; 这个界面就是我们经常看到的登录界面&#xff0c;在这个登录界…

【有奖体验】COS插件体验,四重好礼等你拿!

对象存储 COS 为降低用户接入门槛&#xff0c;集成了多款 COS 插件&#xff0c;开放供用户使用&#xff0c;包含搭建网站、图床、论坛等多个热门业务场景的插件&#xff0c;让使用更便捷&#xff01;对象存储 COS 准备了多重好礼&#xff0c;欢迎广大同学们踊跃体验 COS 插件&a…

U盘安装CentOS7系统出现dracut timeout的解决办法

文章目录 业务场景操作步骤U盘装CentOS7系统确定U盘盘符修改启动命令系统配置 总结 业务场景 我们在某市实施交通信控平台项目&#xff0c;我们申请了一台服务器&#xff0c;用于平台安装由于机房机器只有内网&#xff0c;不连互联网&#xff0c;我们无法安装所需要的软件&…

FreeRTOS(任务通知)

资料来源于硬件家园&#xff1a;资料汇总 - FreeRTOS实时操作系统课程(多任务管理) 目录 一、任务通知的概念 1、概念 2、发送通知给任务的方式 3、任务通知使用限制 二、任务通知的运行机制 三、任务通知的API函数 1、任务通知的数据结构 2、常用的API函数 3、函数x…

Java多线程编程中的线程死锁

Java多线程编程中的线程死锁 ​ 在多线程编程中&#xff0c;线程死锁是一种常见的问题&#xff0c;它发生在两个或多个线程互相等待对方释放资源的情况下&#xff0c;导致程序无法继续执行。本文将介绍线程死锁的概念、产生原因、示例以及如何预防和解决线程死锁问题。 线程死…