Spring 的注入

本文详细介绍了Spring框架中的注入机制,包括为什么需要注入、如何进行Set和构造注入、不同类型变量的处理、构造方法重载以及反转控制和依赖注入的概念。着重讲解了Set注入的详细步骤和构造注入的开发步骤,对比了两者在实际项目中的使用情况。
摘要由CSDN通过智能技术生成

目录

一、注入(Injection)

 1、什么是注入

(1)为什么需要注入

(2)如何进行注入

2、Spring 注入原理分析(简易版)

二、Set 注入详解

1、JDK 内置类型

(1)String + 8种基本类型

(2)数组

(3)Set 集合

(4)List 集合

(5)Map 集合

(6)properties

(7)复杂的 JDK 类型

2、用户自定义类型

第一种方式: 

第二种方式:

3、Set 注入的简化写法

(1)基于属性简化

(2)基于 p 命名空间简化

三、构造注入

1、开发步骤

2、构造方法重载

(1)参数个数不同​编辑

(2)构造参数个数相同时 

3、注入总结

四、反转控制 与 依赖注入

1、反转控制(IoC Inverse of Control)

2、依赖注入(Dependency Injection DI)


一、注入(Injection)

 1、什么是注入

所谓的注入,指的就是,通过 Spring 的工厂及配置文件为所创建对象的成员变量赋值

(1)为什么需要注入

我们原来对一个对象的变量进行赋值使用的是 get 和 set 的方法来完成的:

但是需要注意的是,此时我们所完成的赋值,是我们通过代码来完成的,并不是我们所说的注入

注入指的是通过 Spring 的配置文件对成员变量进行赋值

我们直接通过 set 来进行赋值的效果,是可以达到预期的,但是它并不好

因为它存在耦合,主要体现在

比如,日后如果这个名字不想叫馒头警告了,想叫别的名字,那么这个名字得在代码中更改,实际上这个名字在更改的过程中,就存在耦合了因为改代码涉及到重新编译,重新输出


(2)如何进行注入

1、类的成员变量提供 set,get 方法

2、 配置 Spring 的配置文件

配置文件之前,得先有对应的 get 和 set 方法!!!

此时,就完成了通过配置文件给 id 属性和 name 属性的赋值


2、Spring 注入原理分析(简易版)

 

Spring 通过底层调用对象属性对应的 set 方法来完成成员变量的赋值,这种方式我们也称之为 set 注入


二、Set 注入详解

成员变量的类型,有很多种,我们之前讲的是针对 int 和 String 类型的,所以标签使用的是 value

那如果类型变成了别的呢?比如:double ,char ,List.....

此时 <property> 里面的标签可能就要发生改变了,因为它可能是一个集合,也可能是一个数组,甚至还有可能是一个我们自己定义的类型....

针对于不同类型的成员变量,我们在 <property> 标签中,需要嵌套其它标签

<property>
 xxxxx
</property>

针对不同类型的成员变量,它有两种分类,一种叫做 JDK 类型的变量,也就是 java 原生的成员变量,另一种叫做 用户自定义类型 ,也就是我们自己提供的变量类型

那么接下来,我们就要针对这些不同的类型,来学习在 set 注入过程当中我们应该在 <property> 标签中嵌套什么标签来对对应的变量类型进行赋值


1、JDK 内置类型

(1)String + 8种基本类型

这里所嵌套的标签也就是我们之前所讲的 value 标签

<value>suns</value>

(2)数组

如果现在有一个 数组 类型的成员变量又该如何赋值呢?

        <property name="emails">
            <list>
                <value>123456@qq.com</value>
            </list>
        </property>

(3)Set 集合

当变量类型为 Set 集合时:

       <property name="tels">
                <set>
                    <value>1381111</value>
                    <value>1391111</value>
                    <value>1661111</value>
                </set>
        </property>

这个时候,实际上就是为 set 集合赋值三个元素,因为 set 集合是无序的,所以我们所赋值的这三个元素,虽然我们按照顺序来进行赋值,但是未来在遍历的时候,它有可能不按照这个顺序进行注入

如果此时,这里有一个重复性的数据

比如这个手机号出现了三次,那么按照 set 集合的语义,它会帮我们把重复的数据过滤掉,最终实际上 set 集合在输出的时候,还是之前那三个元素的部分,后面重复的号码被自动过滤掉了

注意:Set 集合中的 Set 标签是固定的,但是里面嵌套的 value 标签却是不一定的

我们此时在 set 里面嵌套的是 value 标签是因为我们定义的泛型是 String ,而  String 是应该用 value,所以我们现在之所以能够嵌套 value 标签使用,原因在于 Person 当中 Set 的泛型是 String 的


(4)List 集合

当类型为 list 时:

 List 集合所使用的 list 标签和我们刚才学的数组一样,都是 list 标签

        <property name="adresses">
            <list>
                <value>shengzhen</value>
                <value>beijing</value>
                <value>shanghai</value>
            </list>
        </property>

List 集合是有序的!!!

所以我们在这里按照这个顺序进行赋值,那么最后 List 集合进行遍历的时候,也一定会按照这个集合进行输出

List 集合是可以重复的!!!

所以即便有三个一样的元素,那么最后输出的时候,这三个元素都会进行输出

和 Set 集合相同的是,我们之所以能在 list 里面嵌套 value 标签,是因为此时我们的泛型是 String 类型的


(5)Map 集合

当类型是 Map 类型时:

我们之前学习 Map 的时候,就已经学习到:Map 是由键值对组成的,那么作为 map 来讲,这个键值对应该封装成一个对象,这个对象我们在之前的学习内容中就已经接触过了:Map.entry

一个 entry ,封装的就是一个键值对 


        <property name="qqs">
            <map>
                <entry>
                    <key>
                        <value>123</value>
                    </key>
                    <value>12415513</value>
                </entry>
            </map>
        </property>

注意:此时在 key 中嵌套 value 也是因为此时的泛型是 String 类型的

同样的,因 为 值 的泛型也是 String,所以后面使用的也是 value


(6)properties

Properties类型 是一个 特殊的Map 特殊的地方在于:key=String value=String
        <property name="p">
            <props>
                <prop key="key1">value1</prop>
                <prop key="key2">value2</prop>
            </props>
        </property>

 这里的每一个 prop 都对应了一个键值对


(7)复杂的 JDK 类型

需要程序员自定义类型转换器,进行处理


2、用户自定义类型

第一种方式: 

1、为成员变量提供 set ,get 方法

2、 在配置文件中进行注入(赋值)   

     <bean id="userService" class="UserServiceImpl">
       <property name="userDAO">
           <bean class="UserDAOImpl"/>
       </property>
    </bean>


第二种方式:

第一种方式存在的一些问题:

1、配置文件代码冗余

2、被注入的对象(userDAO)被多次创建,浪费(JVM)内存资源

第二种方式的步骤与之前一样:

1、为成员变量提供 get ,set 方法

2、配置文件中进行配置(这里有区别)

先创建 userDAO 对象,再来创建 userService 对象,然后<ref bean 引用 userDAO 对象

<bean id="userDAO" class="xxx.UserDAOImpl"/></bean>

<bean id="userService" class="xxx.UserServiceImpl">
     <property name="userDAO">
          <ref bean="userDAO"/>
     </property> 
</bean>

<bean id="orderService" class="xxx.OrderServiceImpl">
     <property name="userDAO">
          <ref bean="userDAO"/>
     </property> 
</bean>


#Spring4.x 废除了 <ref local=""/> 基本等效 <ref bean=""/>

userDAO 被多个引用,解决了代码冗余的问题,同时自始至终只用了一个 bean 标签,创建了一个 DAO 对象,也解决了内存资源浪费的问题


3、Set 注入的简化写法

(1)基于属性简化

JDK类型注⼊
<property name="name">
 <value>suns</value>
</property>
<property name="name" value="suns"/>

注意:value属性 只能简化 8种基本类型+String 注⼊标签

⽤户⾃定义类型
<property name="userDAO">
 <ref bean="userDAO"/>
</property>
<property name="userDAO" ref="userDAO"/>

(2)基于 p 命名空间简化 

JDK类型注⼊
<bean id="person" class="xxxx.Person">
     <property name="name">
         <value>suns</value>
     </property>
</bean>

<bean id="person" class="xxx.Person" p:name="suns"/>

注意:value属性 只能简化 8种基本类型+String 注⼊标签

⽤户⾃定义类型
<bean id="userService" class="xx.UserServiceImpl">
     <property name="userDAO">
         <ref bean="userDAO"/>
     </property>
</bean>

<bean id="userService" class="xxx.UserServiceImpl" p:userDAO-ref="userDAO"/>

三、构造注入

注入:通过 Spring 的配置文件为成员变量赋值

Set 注入:Spring 调用 Set 方法,通过配置文件为成员变量赋值

构造注入:Spring 调用构造方法,为成员变量赋值

1、开发步骤

(1)提供有参构造方法

因为我们得通过有参的构造方法来为对应的成员变量进行赋值

(2)提供Spring 的配置文件进行配置

一个 <constructor-arg> 标签,对应一个构造方法中的构造参数,个数以及顺序和构造参数的顺序是一样的


2、构造方法重载

 当构造方法发生了重载之后,我们要如何进行注入呢?

构造方法重载:方法名一样,但是参数表不                                                                              同

而所谓的参数表不同指的是参数的类型不同,参数的个数不同,参数的顺序不同,那么一旦构造方法发生重载,我们在注入的时候又要注意哪些细节呢?

(1)参数个数不同

通过控制 <constructor-arg> 标签的数量来进行区分 

    <bean id="customer2" class = "Constructor.customer">
        <constructor-arg>
            <value>xiaohei</value>
        </constructor-arg>
    </bean>

(2)构造参数个数相同时 

然后,我们再对 age 进行实验:

    <bean id="customer" class = "Constructor.Customer">
        <constructor-arg>
            <value>18</value>
        </constructor-arg>
    </bean>

此时就发现了一个问题:这个 18 并没有给 age 赋值,而是给 name 赋值了

这是因为 Spring 优先选择了上面的构造方法进行使用

那么如何解决这个方法呢?

name 和 age 最大的区别就是类型不同,所以这个时候我们要额外进行一个限定:指定类型

在  <constructor-arg> 后面新增一个属性,叫做 type

此时就能解决问题了


3、注入总结

在未来写项目的时候,是使用 Set 注入更多还是 构造注入 更多呢?

使用 Set 注入更多

1、构造注入比较麻烦(存在重载的问题)

2、Spring 框架底层,大量应用了 set 注入

综上,在未来,set 注入使用的更多


四、反转控制 与 依赖注入

1、反转控制(IoC Inverse of Control)

控制:对于成员变量赋值的控制权

反转控制:把对于成员变量赋值的控制权,从代码中转移到 Spring 的工厂和配置文件中完成

优点:解耦合

底层实现:工厂设计模式


2、依赖注入(Dependency Injection DI)

注入:通过 Spring 的工厂及配置文件为对象(Bean / 组件)的成员变量赋值

依赖:当一个类需要另一个类的时候,就意味着依赖,一旦出现依赖,就可以把另一个类作为本类的成员变量,最终通过 Spring 进行注入(赋值)

评论 22
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值