主要介绍xml中依赖注入的配置
构造器注入的3种方式详解
set方法注入详解
注入容器中的其他bean的2种方式
其他常见类型注入详解
通常情况下,系统中类和类之间是有依赖关系的,如果一个类对外提供的功能需要通过调用其他类的方法来实现的时候,说明这两个类之间存在依赖关系,如:
public class UserService{
public void insert(UserModel model){
//插入用户信息
}
}
public class UserController{
private UserService userService;
public void insert(UserModel model){
this.userService.insert(model);
}
}
UserController中的insert方法中需要调用userService的insert方法,说明UserController依赖于UserService,如果userService不存在,此时UserControler无法对外提供insert操作。
那么我们创建UserController对象的时候如何将给userService设置值呢?通常有2种方法。
UserController中添加一个有参构造方法,如下:
public class UserController{
private UserService userService;
public UserController(UserService userService){
this.userService = userService;
}
public void insert(UserModel model){
this.userService.insert(model);
}
}
//UserController使用
UserSerivce userService = new UserService();
UserController userController = new UserController(userService);
//然后就可以使用userController对象了
可以在UserController中给userService添加一个set方法,如:
public class UserController{
private UserService userService;
public setUserService(UserService userService){
this.userService = userService;
}
public void insert(UserModel model){
this.userService.insert(model);
}
}
//UserController使用
UserSerivce userService = new UserService();
UserController userController = new UserController();
userController.setService(userService);
//然后就可以使用userController对象了
上面这些操作,将被依赖的对象设置到依赖的对象中,spring容器内部都提供了支持,这个在spirng中叫做依赖注入。
spring中依赖注入主要分为手动注入和自动注入,本文我们主要说一下手动注入,手动注入需要我们明确配置需要注入的对象。
刚才上面我们回顾了,将被依赖方注入到依赖方,通常有2种方式:构造函数的方式和set属性的方式,spring中也是通过这两种方式实现注入的,下面详解2种方式。
构造器的参数就是被依赖的对象,构造器注入又分为3种注入方式:
根据构造器参数索引注入
根据构造器参数类型注入
根据构造器参数名称注入
constructor-arg用户指定构造器的参数
index:构造器参数的位置,从0开始
value:构造器参数的值,value只能用来给简单的类型设置值,value对应的属性类型只能为byte,int,long,float,double,boolean,Byte,Long,Float,Double,枚举,spring容器内部注入的时候会将value的值转换为对应的类型。
package com.javacode2018.lesson001.demo5;
public class UserModel {
private String name;
private int age;
//描述信息
private String desc;
public UserModel() {
}
public UserModel(String name, String desc) {
this.name = name;
this.desc = desc;
}
public UserModel(String name, int age, String desc) {
this.name = name;
this.age = age;
this.desc = desc;
}
@Override
public String toString() {
return "UserModel{" +
"name='" + name + '\'' +
", age=" + age +
", desc='" + desc + '\'' +
'}';
}
}
注意上面的3个构造函数。
上面创建UserModel实例代码相当于下面代码:
UserModel userModel = new UserModel("路人甲Java","我是通过构造器参数类型注入的");
package com.javacode2018.lesson001.demo5;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!
*/
public class IocUtil {
public static ClassPathXmlApplicationContext context(String beanXml) {
return new ClassPathXmlApplicationContext(beanXml);
}
}
package com.javacode2018.lesson001.demo5;
import org.junit.Test;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
* 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!
*/
public class DiTest {
/**
* 通过构造器的参数位置注入
*/
@Test
public void diByConstructorParamIndex() {
String beanXml = "classpath:/com/javacode2018/lesson001/demo5/diByConstructorParamIndex.xml";
ClassPathXmlApplicationContext context = IocUtils.context(beanXml);
System.out.println(context.getBean("diByConstructorParamIndex"));
}
}
运行diByConstructorParamIndex输出
UserModel{name='路人甲Java', age=0, desc='我是通过构造器参数位置注入的'}
参数位置的注入对参数顺序有很强的依赖性,若构造函数参数位置被人调整过,会导致注入出错。
不过通常情况下,不建议去在代码中修改构造函数,如果需要新增参数的,可以新增一个构造函数来实现,这算是一种扩展,不会影响目前已有的功能。
constructor-arg用户指定构造器的参数
type:构造函数参数的完整类型,如:java.lang.String,int,double
value:构造器参数的值,value只能用来给简单的类型设置值
上面创建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) {
}
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 + '\'' +
'}';
}
}
上面创建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='我是通过构造器参数类型注入的'}
通常情况下,我们的类都是标准的javabean,javabean类的特点:
属性都是private访问级别的
属性通常情况下通过一组setter(修改器)和getter(访问器)方法来访问
setter方法,以set开头,后跟首字母大写的属性名,如:setUserName,简单属性一般只有一个方法参数,方法返回值通常为void;
getter方法,一般属性以get开头,对于boolean类型一般以is开头,后跟首字母大写的属性名,如:getUserName,isOk;
spring对符合javabean特点类,提供了setter方式的注入,会调用对应属性的setter方法将被依赖的对象注入进去。
...
property用于对属性的值进行配置,可以有多个
name:属性的名称
value:属性的值
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 +
'}';
}
}
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有两种写法:
ref属性方式
内置bean的方式
将上面介绍的constructor-arg或者property元素的value属性名称替换为ref,ref属性的值为容器中其他bean的名称,如:
构造器方式,将value替换为ref:
setter方式,将value替换为ref:
构造器的方式:
setter方式:
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配置文件
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=''}}
Spring
或
或
或
或
或
数组中的元素
Properties类相当于键值都是String类型的Map对象,使用props进行注入,如下:
java高并发系列
mybatis系列
mysql系列
对于上面这些类型来个综合案例。
package com.javacode2018.lesson001.demo5;
import java.util.*;
public class DiOtherTypeModel {
private List list1;
private Set set1;
private Map 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 getMap1() {
return map1;
}
public void setMap1(Map 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 +
'}';
}
}
上面这个类中包含了刚才我们介绍的各种类型,下面来进行注入。
Spring
SpringBoot
10
9
8
java高并发系列
mybatis系列
mysql系列
DiTest类中新增一个方法
/**
* 其他各种类型注入
*/
@Test
public void diOtherType() {
String beanXml = "classpath:/com/javacode2018/lesson001/demo5/diOtherType.xml";
ClassPathXmlApplicationContext context = IocUtils.context(beanXml);
System.out.println(context.getBean("diOtherType"));
}
运行测试用例输出:
DiOtherTypeModel{list1=[Spring, SpringBoot], set1=[UserModel{name='null', age=0, desc='null'}, UserModel{name='null', age=0, desc='null'}], map1={路人甲Java=30, 路人=28}, array1=[10, 9, 8], properties1={key3=mysql系列, key2=mybatis系列, key1=java高并发系列}}
本文主要讲解了xml中bean的依赖注入,都是采用硬编码的方式进行注入的,这种算是手动的方式
注入普通类型通过value属性或者value元素设置注入的值;注入对象如果是容器的其他bean的时候,需要使用ref属性或者ref元素或者内置bean元素的方式
还介绍了其他几种类型List、Set、Map、数组、Properties类型的注入,多看几遍加深理解
后面我们将介绍spring为我们提供的更牛逼的自动注入
链接:https://pan.baidu.com/s/1p6rcfKOeWQIVkuhVybzZmQ
提取码:zr99
Spring系列第1篇:为何要学spring?
Spring系列第2篇:控制反转(IoC)与依赖注入(DI)
Spring系列第3篇:Spring容器基本使用及原理
Spring系列第4篇:xml中bean定义详解(-)
Spring系列第5篇:创建bean实例这些方式你们都知道?
Spring系列第6篇:玩转bean scope,避免跳坑里!
Java高并发系列(共34篇)
MySql高手系列(共27篇)
Maven高手系列(共10篇)
Mybatis系列(共12篇)
聊聊db和缓存一致性常见的实现方式
接口幂等性这么重要,它是什么?怎么实现?
泛型,有点难度,会让很多人懵逼,那是因为你没有看这篇文章!
感谢大家的阅读,也欢迎您把这篇文章分享给更多的朋友一起阅读!谢谢!
路人甲java
▲长按图片识别二维码关注
路人甲Java:工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!