基于XML的自动装配之场景模拟:
自动装配
:根据指定的策略,在IOC容器中匹配某一个bean,自动为指定的bean中所依赖的类类型或者接口类型赋值
之前我们学过的依赖注入,我们在为不同属性赋值时,例如类类型的属性,我们使用的是ref为其赋值,而字面量我们使用的是value,而对于类类型来说,我们只需要找到IOC容器中所对应的bean的对象,通过ref属性来引用该bean所对应的id,我们就把该bean所对应的对象来为类类型的属性或者接口类型的属性赋值,那么自动装配使得我们不再需要通过书写property去指定某一个属性,再通过当前的ref去引用某一个id了,我们设置了自动装配之后,那么当前这个bean所有的类类型的属性都可以找到所匹配的bean来自动为当前属性赋值。
那么下面我们就来进行一个场景的模拟:其包含三层架构包括:控制层、业务层、持久层
创建完成之后,通过代码来处理他们之间的关系:
第一步:在UserController中创建service对象:
public class UserController {
private UserService userService=new UserServiceImpl();
}
上述这种方式是我们没有学习IOC之前的写法,它的缺点在于,功能不是单一且一成不变的,若此后service接口有新的实现类,或者是我们想对当前类进行更新和维护,如果我们还是上述这种写法,那么只能在原有的代码上进行修改,以便于扩展新的实现类等,修改代码之后,代码需要重新编译,打包等,这样才能呈现出我们修改之后的效果
但学习了IOC之后,我们知道IOC是用来管理对象和对象之间的依赖关系
,那么我们就可以将UserController交给IOC容器管理,UserServiceImpl,UserDaoImpl也是可以交给IOC容器管理的,注意,这里说的是接口的实现类,并不是接口,因为我们设置一个bean标签,它的class只能是一个类,而不能是一个接口
将这三个组件交给IOC容器管理之后,那么就可以在UserController中设置Userservice的成员变量,并且设置该成员变量的set和get方法,由于我们将UserController交给IOC容器管理,那么Userservice就会被动的接受IOC容器的注入,也就是说我们可以通过set为当前的接口赋值,现在Userservice的实现类是UserServiceImpl,那么此后如果有新的实现类,我们只需要在配置文件中,将其id修改为新的实现类的id即可
其正确写法如下所示:
package Controller;
import Service.UserService;
public class UserController {
private UserService userService ;
public UserService getUserService() {
return userService;
}
public void setUserService(UserService userService) {
this.userService = userService;
}
}
第二步:处理UserServiceImpl类
package Service.impl;
import Service.UserService;
import dao.UserDao;
public class UserServiceImpl implements UserService {
//设置Dao并为其设置set和get方法或者设置有参构造,因为当我们将对象交给IOC容器管理之后,我们要想对其进行赋值,要么使用set注入,要么是构造器注入
public UserDao userDao;
public UserDao getUserDao() {
return userDao;
}
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
}
假设我们在Controller类
中加入下述方法,模拟实现调用userservice中的save方法来处理业务逻辑:
public void save(){
userService.save();
}
但此时save方法在UserService接口并不存在啊,因此我们需要在UserService 接口中去创建该方法 ,如下所示:
//保存用户信息
void save();
由于UserServiceImpl实现了UserService接口,那么需要对其方法进行实现,通过userDao去调用,如下所示在UserServiceImpl
中添加下述代码:
@Override
public void save() {
userDao.save();
}
同样的UserDao中并没有该方法,因此我们也需要在该接口中去创建该方法,如下所示在UserDao
中添加下述代码:
void save();
由于UserDaoImpl
实现了UserDao接口,因此也需要实现该方法,将该方法重写如下所示:
@Override
public void save() {
System.out.println("保存成功");
}
下面就可以将其三层架构以及之间的逻辑关系交给IOC容器管理:
三层架构的三个组件交给IOC容器管理,其实并不仅仅是将这三个组件交给IOC容器管理,还将它们之间的依赖关系交给IOC容器管理
在resources下创建,spring-autowrie-xml.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标签中的class一定是一个类,而不是接口,接口类型的这里一定写的是其实现类-->
<!-- 控制层中来调用service处理业务逻辑-->
<bean id="UserController" class="Controller.UserController">
<property name="userService" ref="userService"></property>
</bean>
<!--service中调用Dao实现持久化操作-->
<bean id="userService" class="Service.impl.UserServiceImpl">
<property name="userDao" ref="userDao"></property>
</bean>
<bean id="userDao" class="dao.impl.UserDaoImpl"></bean>
</beans>
编写测试类
:
import Controller.UserController;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class autowireTest {
@Test
public void test(){
ApplicationContext ioc=new ClassPathXmlApplicationContext("spring-autowrie-xml.xml");
UserController userController=ioc.getBean(UserController.class);
userController.save();
}
}
输出如下所示:
保存成功
基于XML的自动装配之byType:
上面的场景模拟中,我们依然是通过property标签手动的配置,但当我们设置了自动装配之后,就不需要通过property标签手动赋值,只要配置自动装配,它就可以自动在IOC容器中找到某一个bean,自动为当前属性赋值,但需要注意的是,它只针对于类类型或者接口类型,而如果是字面量类型的属性,是不可以的
既然如此,自动装配使得我们免于书写property标签,那么我们是否可以将其property标签删除呢?
测试结果如下:
编译器报了一个空指针异常,由于我们上述的操作是将赋值的代码块注释了,此时的bean标签中的 Controller的UserService就是默认值,即为null
由此接下来,我们将学习如何配置自动装配:
修改spring-autowrie-xml.xml文件中的bean标签
:
通过设置autowire的值为byType,相当于设置了自动装配,当前的bean对象可通过要赋值属性的值,在IOC中匹配某个bean,为属性赋值
<bean id="UserController" class="Controller.UserController" autowire="byType"></bean>
<bean id="userService" class="Service.impl.UserServiceImpl" autowire="byType"></bean>
<bean id="userDao" class="dao.impl.UserDaoImpl"></bean>
测试依然成功:
但如果我们在当前的IOC容器一个bean都无法匹配到呢?
修改spring-autowrie-xml.xml文件,使其只保留一个bean对象
,测试结果如下:
依然是出现空指针异常,也就说明了当我们把autowire属性的值设置为byType时,如果能够匹配到bean,那么就为其赋值,否则就使用默认值
如下所示,如果当在一个IOC容器中同时能够匹配到两个bean时,未测试之前,编译器都报错,
<bean id="UserController" class="Controller.UserController" autowire="byType"></bean>
<bean id="userService" class="Service.impl.UserServiceImpl" autowire="byType"></bean>
<bean id="Service" class="Service.impl.UserServiceImpl" autowire="byType"></bean>
<bean id="userDao" class="dao.impl.UserDaoImpl"></bean>
<bean id="Dao" class="dao.impl.UserDaoImpl"></bean>
测试结果如下:
当通过类型找到了多个类型匹配的bean,会抛出NoUniqueBeanDefinitionException[没有唯一的bean被匹配]异常
当使用byType实现自动装配时,IOC容器中有且只有一个类型的bean能够为属性赋值