注入自定义对象
上一篇文章说到了注入java内置对象的方法(Setter) 这次来说一说关于注入自定义类型对象的一些内容
Bean和注入
定义两个Bean如下所示:
public class User {
private String name;
private String sex;
private Role role;
}
public class Role {
private String role_name;
}
两个Bean里面都实现Get()和Set()方法 并且重写toString()方法
xml主配置文件内使用如下配置进行测试:
<bean id="user" class="com.test.model.User">
<property name="name" value="wukong" />
<property name="sex" value="男" />
<!-- 此时name所指代的属性 不再是简单的数据类型 而是自定义的对象 -->
<!-- 再次使用bean标签来进行实例化 -->
<property name="role">
<bean id="role" class="com.test.model.Role">
<property name="role_name" value="斗战胜佛" />
</bean>
</property>
</bean>
在测试文件内(使用Junit)编写测试方法:
@Test
public void GetUserMessige(){
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
//User指代xml文件内的bean标签的id
//User.class指明User的类路径 利用框架映射帮助我们进行实例化
User user= ac.getBean("user",User.class);
System.out.println(user);
}
控制台输出:
user [name=wukong, sex=男, role=role [role_name=斗战胜佛]]
但是这样的写法 会存在这样一个大问题:
我们在xml文件内定义了两个<bean> id分别为 user 和 role
但是我们只能获取到id为user的bean内的内容,role我们获取不到
进行如下测试:
xml配置文件内容保持上述不动 在Junit测试文件内新建测试方法 指向id为role的bean标签
@Test
public void GetRoleMessige(){
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
Role role= ac.getBean("role",Role.class);
System.out.println(role);
}
进行测试发现Junit报错:
报错内容为 : No bean named 'role' available
没有名为role的bean被定义
出现这个报错的原因很好理解
==>
我们的确是定义了两个bean
但是它们是包含与被包含的关系
role在user的内层 其作用范围在uesr内
我们发出了想要越过user来直接获取内层bean的请求
自然是获取不到
因此 我们在内层的bean上设置id 其实没有任何意义
综上 我们还有第二种解决方案
我们将role的配置单独书写 并将两个bean建立联系
代码如下:
<bean id="user" class="com.test.model.User">
<property name="name" value="wukong" />
<property name="sex" value="男" />
<property name="role">
<!-- 使用<ref>标签来使两个独立的bean建立联系 -->
<ref bean="role"/>
</property>
</bean>
<bean id="role" class="com.test.model.Role">
<property name="role_name" value="斗战胜佛" />
</bean>
</beans>
存在这样一种简写方式:
<property name="role" ref="role"/>
<!--
name="role" 指代类中的属性
ref="role" 指代独立的bean的id值
-->
可以将上面配置联系的代码进行简化书写代码
接口和注入
在开发当中我们经常会使用到接口 按照传统的开发模式 现在我们有了新的包结构 如下图:
在正常的开发模式下 我们需要在接口的实现类下进行接口回调 如下图
现在我们可以尝试通过注入的方式来完成操作
在xml文件内声明两个实现类:
<bean id="userDaoImpl" class="com.test.dao.impl.UserDaoImpl"/>
<bean id="userServiceImpl" class="com.test.service.impl.UserServiceImpl">
<!--
UserDaoImpl userdaoimpl = new UserDaoImpl();
UserDao userDao = userdaoimpl;
-->
<property name="userdao" ref="userDaoImpl"></property>
</bean>
可以看到 我们通过注入方式 利用框架帮助我们完成了接口回调的操作 具体实现逻辑 看上述代码的注释部分
此时 我们可以直接通过UsereService进行测试了:
@Test
public void 测试UserService(){
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
UserService userService= ac.getBean("userServiceImpl",UserService.class);
userService.Test01();
}
注意:
在书写这段代码的时候 ac.getBean("userServiceImpl",UserService.class);
双引号括起来的地方 指向Service层bean的id
后面的类路径 需要书写接口的类路径 不是接口实现类(impl)的类路径
代码提示:其实本段代码 类似于上面所列出的User和Role的例子 不过是把实体bean转换成了接口及其实现类 一点都不复杂 千万别迷糊 hhhhh
构造函数注入方式
在前文 我们主要使用Setter注入方式 类中只有存在set方法才能够使用
下面我们来测试另一种注入方式:构造函数注入
我们更改User类里面的内容 如下:
private String name;
private String sex;
private Role role;
public User(){
System.out.println("构造方法");
}
public void setName( String name ) {
System.out.println("setName");
this.name = name;
}
public void setSex( String sex ) {
System.out.println("setSex");
this.sex = sex;
}
public void setRole( Role role ) {
System.out.println("setRole");
this.role = role;
}
然后重写toString()方法
重新进行测试 结果如下:
可以看到具体的流程:
现在我们所写的bean标签的内容 是基于java默认为每一个类提供的无参的构造函数
每次加载bean标签 都会首先去寻找所引用的类中的构造方法 然后通过公开的Set()方法 往实例化好的对象里面添加值
所以我们每个类当中 都要存在一个无参的构造方法来进行控制反转
现在我们添加一个有参的构造函数:
public User( String name , String sex ) {
super();
this.name = name;
this.sex = sex;
}
在xml配置文件内添加如下代码:
<bean id="user1" class="com.test.model.User">
<constructor-arg name="name" value="八戒"/>
<constructor-arg name="sex" value="男"/>
</bean>
新建bean标签做注入 <constructor-arg>标签对应类中的构造函数
因为在User类中的有参构造函数有两个参数 所以<constructor-arg>标签要写两个 分别对应两个参数 并给予初始值
此时新建测试方法:
@Test
public void 测试User构造函数(){
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
User user= ac.getBean("user1",User.class);
System.out.println(user);
控制台输出:
产生 构造方法 SetName 等结果
是因为我们同一个xml文件中配置了使用无参构造函数 使用setter方式进行注入
xml文件一被加载 内部所有的bean都会被进行实例化
别惊慌hhhh
值得注意的是 在 <constructor-arg>标签内
存在index属性 用来标识形参的位置 起始值为1
存在type属性 指明参数类型
如刚才我们对构造函数进行的操作,完整内容应为如下:
<bean id="user1" class="com.test.model.User">
<constructor-arg name="name" index="0" type="java.lang.String" value="八戒"/>
<constructor-arg name="sex" index="1" type="java.lang.String" value="男"/>
</bean>
关于构造函数注入自定义类型 类似于下面的语句就可以完成:
指明类中属性 指明引用的bean(我们在上文曾新建了一个专门用来实例化Role类型的bean)
<constructor-arg name="role" ref="role" />