Spring系列第7篇:依赖注入之手动注入

constructor-arg用户指定构造器的参数

type:构造函数参数的完整类型,如:java.lang.String,int,double

value:构造器参数的值,value只能用来给简单的类型设置值

案例
diByConstructorParamType.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-4.3.xsd">

上面创建UserModel实例代码相当于下面代码:

UserModel userModel = new UserModel(“路人甲Java”,30,“我是通过构造器参数类型注入的”);

新增测试用例

DiTest类中新增一个测试用例

/**

* 通过构造器的参数类型注入

*/

@Test

public void diByConstructorParamType() {

String beanXml = “classpath:/com/javacode2018/lesson001/demo5/diByConstructorParamType.xml”;

ClassPathXmlApplicationContext context = IocUtils.context(beanXml);

System.out.println(context.getBean(“diByConstructorParamType”));

}

效果

运行diByConstructorParamType输出

UserModel{name=‘路人甲Java’, age=30, desc=‘我是通过构造器参数类型注入的’}

优缺点

实际上按照参数位置或者按照参数的类型注入,都有一个问题,很难通过bean的配置文件,知道这个参数是对应UserModel中的那个属性的,代码的可读性不好,比如我想知道这每个参数对应UserModel中的那个属性,必须要去看UserModel的源码,下面要介绍按照参数名称注入的方式比上面这2种更优秀一些。

根据构造器参数名称注入

用法

constructor-arg用户指定构造器的参数

name:构造参数名称

value:构造器参数的值,value只能用来给简单的类型设置值

关于方法参数名称的问题

java通过反射的方式可以获取到方法的参数名称,不过源码中的参数通过编译之后会变成class对象,通常情况下源码变成class文件之后,参数的真实名称会丢失,参数的名称会变成arg0,arg1,arg2这样的,和实际参数名称不一样了,如果需要将源码中的参数名称保留在编译之后的class文件中,编译的时候需要用下面的命令

javac -parameters java源码

但是我们难以保证编译代码的时候,操作人员一定会带上-parameters参数,所以方法的参数可能在class文件中会丢失,导致反射获取到的参数名称和实际参数名称不符,这个我们需要先了解一下。

参数名称可能不稳定的问题,spring提供了解决方案,通过ConstructorProperties注解来定义参数的名称,将这个注解加在构造方法上面,如下:

@ConstructorProperties({“第一个参数名称”, “第二个参数的名称”,…“第n个参数的名称”})

public 类名(String p1, String p2…,参数n) {

}

案例
CarModel.java

package com.javacode2018.lesson001.demo5;

import java.beans.ConstructorProperties;

public class CarModel {

private String name;

//描述信息

private String desc;

public CarModel() {

}

@ConstructorProperties({“name”, “desc”})

public CarModel(String p1, String p2) {

this.name = p1;

this.desc = p2;

}

@Override

public String toString() {

return “CarModel{” +

“name='” + name + ‘’’ +

“, desc='” + desc + ‘’’ +

‘}’;

}

}

diByConstructorParamName.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-4.3.xsd">

上面创建CarModel实例代码相当于下面代码:

CarModel carModel = new CarModel(“保时捷Macans”,“我是通过构造器参数类型注入的”);

新增测试用例

DiTest类中新增一个测试用例

/**

* 通过构造器的参数名称注入

*/

@Test

public void diByConstructorParamName() {

String beanXml = “classpath:/com/javacode2018/lesson001/demo5/diByConstructorParamName.xml”;

ClassPathXmlApplicationContext context = IocUtils.context(beanXml);

System.out.println(context.getBean(“diByConstructorParamName”));

}

效果

运行diByConstructorParamName输出

CarModel{name=‘保时捷Macans’, desc=‘我是通过构造器参数类型注入的’}

setter注入

通常情况下,我们的类都是标准的javabean,javabean类的特点:

  • 属性都是private访问级别的

  • 属性通常情况下通过一组setter(修改器)和getter(访问器)方法来访问

  • setter方法,以set开头,后跟首字母大写的属性名,如:setUserName,简单属性一般只有一个方法参数,方法返回值通常为void;

  • getter方法,一般属性以get开头,对于boolean类型一般以is开头,后跟首字母大写的属性名,如:getUserName,isOk;

spring对符合javabean特点类,提供了setter方式的注入,会调用对应属性的setter方法将被依赖的对象注入进去。

用法

property用于对属性的值进行配置,可以有多个

name:属性的名称

value:属性的值

案例
MenuModel.java

package com.javacode2018.lesson001.demo5;

/**

* 菜单类

*/

public class MenuModel {

//菜单名称

private String label;

//同级别排序

private Integer theSort;

public String getLabel() {

return label;

}

public void setLabel(String label) {

this.label = label;

}

public Integer getTheSort() {

return theSort;

}

public void setTheSort(Integer theSort) {

this.theSort = theSort;

}

@Override

public String toString() {

return “MenuModel{” +

“label='” + label + ‘’’ +

“, theSort=” + theSort +

‘}’;

}

}

diBySetter.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-4.3.xsd">

新增测试用例

DiTest类中添加一个测试方法

/**

* 通过setter方法注入

*/

@Test

public void diBySetter() {

String beanXml = “classpath:/com/javacode2018/lesson001/demo5/diBySetter.xml”;

ClassPathXmlApplicationContext context = IocUtils.context(beanXml);

System.out.println(context.getBean(“diBySetter”));

}

效果

运行diBySetter输出

MenuModel{label=‘spring系列’, theSort=null}

优缺点

setter注入相对于构造函数注入要灵活一些,构造函数需要指定对应构造函数中所有参数的值,而setter注入的方式没有这种限制,不需要对所有属性都进行注入,可以按需进行注入。

上面介绍的都是注入普通类型的对象,都是通过value属性来设置需要注入的对象的值的,value属性的值是String类型的,spring容器内部自动会将value的值转换为对象的实际类型。

若我们依赖的对象是容器中的其他bean对象的时候,需要用下面的方式进行注入。

注入容器中的bean

注入容器中的bean有两种写法:

  • ref属性方式

  • 内置bean的方式

ref属性方式

将上面介绍的constructor-arg或者property元素的value属性名称替换为ref,ref属性的值为容器中其他bean的名称,如:

构造器方式,将value替换为ref:

setter方式,将value替换为ref:

内置bean的方式

构造器的方式:

setter方式:

案例
PersonModel.java

package com.javacode2018.lesson001.demo5;

public class PersonModel {

private UserModel userModel;

private CarModel carModel;

public PersonModel() {

}

public PersonModel(UserModel userModel, CarModel carModel) {

this.userModel = userModel;

this.carModel = carModel;

}

public UserModel getUserModel() {

return userModel;

}

public void setUserModel(UserModel userModel) {

this.userModel = userModel;

}

public CarModel getCarModel() {

return carModel;

}

public void setCarModel(CarModel carModel) {

this.carModel = carModel;

}

@Override

public String toString() {

return “PersonModel{” +

“userModel=” + userModel +

“, carModel=” + carModel +

‘}’;

}

}

PersonModel中有依赖于2个对象UserModel、CarModel,下面我们通过spring将UserModel和CarModel创建好,然后注入到PersonModel中,下面创建bean配置文件

diBean.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-4.3.xsd">

新增测试用例

DiTest中新增一个测试方法

@Test

public void diBean(){

String beanXml = “classpath:/com/javacode2018/lesson001/demo5/diBean.xml”;

ClassPathXmlApplicationContext context = IocUtils.context(beanXml);

System.out.println(context.getBean(“diBeanByConstructor”));

System.out.println(context.getBean(“diBeanBySetter”));

}

效果

运行diBean用例,输出:

PersonModel{userModel=UserModel{name=‘null’, age=0, desc=‘null’}, carModel=CarModel{name=‘宾利’, desc=‘’}}

PersonModel{userModel=UserModel{name=‘null’, age=0, desc=‘null’}, carModel=CarModel{name=‘保时捷’, desc=‘’}}

其他类型注入

注入java.util.List(list元素)

Spring

注入java.util.Set(set元素)

Spring

注入java.util.Map(map元素)

value对应的值,可以为任意类型

value对应的值,可以为任意类型

注入数组(array元素)

数组中的元素

注入java.util.Properties(props元素)

Properties类相当于键值都是String类型的Map对象,使用props进行注入,如下:

java高并发系列

mybatis系列

mysql系列

案例

对于上面这些类型来个综合案例。

DiOtherTypeModel.java

package com.javacode2018.lesson001.demo5;

import java.util.*;

public class DiOtherTypeModel {

private List list1;

private Set set1;

private Map<String, Integer> map1;

private int[] array1;

private Properties properties1;

public List getList1() {

return list1;

}

public void setList1(List list1) {

this.list1 = list1;

}

public Set getSet1() {

return set1;

}

public void setSet1(Set set1) {

this.set1 = set1;

}

public Map<String, Integer> getMap1() {

return map1;

}

public void setMap1(Map<String, Integer> map1) {

this.map1 = map1;

}

public int[] getArray1() {

return array1;

}

public void setArray1(int[] array1) {

this.array1 = array1;

}

public Properties getProperties1() {

return properties1;

}

public void setProperties1(Properties properties1) {

this.properties1 = properties1;

}

@Override

public String toString() {

return “DiOtherTypeModel{” +

“list1=” + list1 +

“, set1=” + set1 +

“, map1=” + map1 +

“, array1=” + Arrays.toString(array1) +

“, properties1=” + properties1 +

‘}’;

}

}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值