[关闭]
@martin0207 2018-04-23T12:18:50.000000Z 字数 6229 阅读 824

Spring学习——IOC由xml实现

Java学习


Spring框架的基础模块,包含IOC(注入)和AOP(面向切面)。

基本配置使用

基础配置

src文件夹下,创建xml文件,这个文件放置Spring的配置。一般使用beans.xml做名字

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.1.xsd">
  6. </beans>

创建需要注入的类

  1. public class HelloIoc {
  2. public void showMsg() {
  3. System.out.println("测试成功");
  4. }
  5. }

在beans.xml文件中添加配置

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.1.xsd">
  6. <!--
  7. id:获取该注入类时,使用的字段
  8. class:该注入类的全部路径
  9. -->
  10. <bean id="helloIoc" class="model.HelloIoc"></bean>
  11. </beans>

创建测试类及测试方法

  1. public class TestIoc {
  2. /**
  3. * 注入类,统一使用BeanFactory来获取。
  4. * BeanFactory的创建,则是通过以下方式
  5. */
  6. private BeanFactory factory = new ClassPathXmlApplicationContext("beans.xml");
  7. @Test
  8. public void testIocForXml() {
  9. HelloIoc helloIoc = factory.getBean("helloIoc",HelloIoc.class);
  10. helloIoc.showMsg();
  11. }
  12. }

测试结果

  1. 测试成功

单例多例转换

Java中涉及到单例模式与多例模式,注入也会在这两种模式中取舍。

默认模式

首先,我们测试一下默认的模式。创建两个对象,对比两个对象是否相同。

  1. public class TestIoc {
  2. /**
  3. * 注入类,统一使用BeanFactory来获取。
  4. * BeanFactory的创建,则是通过以下方式
  5. */
  6. private BeanFactory factory = new ClassPathXmlApplicationContext("beans.xml");
  7. @Test
  8. public void testIocForXml() {
  9. HelloIoc helloIoc = factory.getBean("helloIoc",HelloIoc.class);
  10. //获取第二个对象
  11. HelloIoc helloIoc2 = factory.getBean("helloIoc",HelloIoc.class);
  12. helloIoc.showMsg();
  13. //对比两个对象
  14. System.out.println(helloIoc==helloIoc2);
  15. }
  16. }

运行结果

  1. 测试成功
  2. true

由此可见,默认模式下,获取的对象是单例。至于单例与多例的区别,这里不做赘述,有需要的小伙伴可以搜一搜。

设置多例/单例

在beans.xml配置文件中,bean标签下有个scope属性,是设置单例/多例模式的,修改配置如下:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.1.xsd">
  6. <!--
  7. id:获取该注入类时,使用的字段
  8. class:该注入类的全部路径
  9. -->
  10. <!--
  11. prototype:多例模式
  12. singleton:单例模式(默认)
  13. -->
  14. <bean id="helloIoc" class="model.HelloIoc" scope="prototype"></bean>
  15. </beans>

运行结果

  1. 测试成功
  2. false

变量注入

在一个类中,我们一般会使用到许多变量,那么我们学习一下这些变量的注入。

创建几个功能类

User类

  1. public class User {
  2. private String name;
  3. private int id ;
  4. public User(int id,String name) {
  5. this.id = id;
  6. this.name = name;
  7. }
  8. public String getName() {
  9. return name;
  10. }
  11. public void setName(String name) {
  12. this.name = name;
  13. }
  14. public int getId() {
  15. return id;
  16. }
  17. public void setId(int id) {
  18. this.id = id;
  19. }
  20. }

UserDao类

  1. public class UserDao {
  2. public void add(User user) {
  3. System.out.println("add " + user.getName());
  4. }
  5. public void delete(int id) {
  6. System.out.println("delete " + id);
  7. }
  8. }

UserService类

  1. public class UserService {
  2. private UserDao dao;
  3. public void add(User user) {
  4. dao.add(user);
  5. }
  6. public void delete(int id) {
  7. dao.delete(id);
  8. }
  9. public UserDao getDao() {
  10. return dao;
  11. }
  12. public void setDao(UserDao dao) {
  13. this.dao = dao;
  14. }
  15. }

因为User类存在变量,我们先不对User注入。对UserDao和UserService的注入如下:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://www.springframework.org/schema/beans
  5. http://www.springframework.org/schema/beans/spring-beans-4.1.xsd">
  6. <!--
  7. id:获取该注入类时,使用的字段
  8. class:该注入类的全部路径
  9. -->
  10. <!--
  11. prototype:多例模式
  12. singleton:单例模式(默认)
  13. -->
  14. <bean id="helloIoc" class="model.HelloIoc" scope="prototype"></bean>
  15. <bean id="userDao" class="dao.UserDao"></bean>
  16. <bean id="userService" class="service.UserService">
  17. <!--
  18. name中的值会在userService对象中调用setXX方法来注入
  19. 诸如:name="dao" ,在具体注入时,setDao(UserDao userDao) 完成注入
  20. ref="userDao" 表示配置文件中,bean中创建的id
  21. -->
  22. <property name="dao" ref="userDao"/>
  23. </bean>
  24. </beans>

测试

写测试方法

  1. @Test
  2. public void testIoc2() {
  3. UserService service = factory.getBean("userService",UserService.class);
  4. User user = new User(1, "测试对象");
  5. service.setUser(user);
  6. service.add();
  7. service.delete(2);
  8. }

运行结果

  1. add 测试对象
  2. delete 2

测试结果出来,再叙述一下:当要注入的类中存在变量时(目前变量依然是注入类),我们可以使用property标签来实现注入:

  1. <bean id="userService" class="service.UserService">
  2. <!--
  3. name中的值会在userService对象中调用setXX方法来注入
  4. 诸如:name="dao" ,在具体注入时,setDao(UserDao userDao) 完成注入
  5. ref="userDao" 表示配置文件中,bean中创建的id
  6. -->
  7. <property name="dao" ref="userDao"/>
  8. </bean>

实现非注入类的变量注入

User类中,String和int型变量都不是注入类,那么我们不能使用ref的方式,一般设置固定值时,都会使用value:

  1. <bean id="user" class="model.User" scope="prototype">
  2. <property name="name" value="注入名"></property>
  3. <property name="id" value="11"></property>
  4. </bean>

在UserService的bean配置中添加配置:

  1. <bean id="userService" class="service.UserService">
  2. <property name="dao" ref="userDao"/>
  3. <property name="user" ref="user"></property>
  4. </bean>

测试

因为这里已经注入,所以讲测试方法中User的创建与赋值去掉:

  1. @Test
  2. public void testIoc2() {
  3. UserService service = factory.getBean("userService",UserService.class);
  4. // User user = new User(1, "测试对象");
  5. // service.setUser(user);
  6. service.add();
  7. service.delete(2);
  8. }

运行一下测试方法,哇,出错了。

  1. org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'userService' defined in class path resource [beans.xml]: Cannot resolve reference to bean 'user' while setting bean property 'user'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'user' defined in class path resource [beans.xml]: Instantiation of bean failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [model.User]: No default constructor found; nested exception is java.lang.NoSuchMethodException: model.User.<init>()

这么一大堆错误信息,还只是第一行,其实只在说一件事:没有发现默认构造函数

即:注入类必须有一个无参构造函数

我们在User类中添加无参构造,再运行:

  1. add 注入名
  2. delete 2

自动注入

每个变量都需要配置bean标签,实在是麻烦透顶,那么你一定动小心思:如果能够自动注入就好了。
是了,Spring开发团队也想到了:autowire标签

再次修改UserService的bean配置:

  1. <bean id="userService" class="service.UserService" autowire="byType">
  2. <!--
  3. <property name="dao" ref="userDao"/>
  4. <property name="user" ref="user"/>
  5. -->
  6. </bean>

测试结果

  1. add 注入名
  2. delete 2

如果这时候使用 byName,则会报错。因为在UserService中,UserDao变量的名称是dao,找不到相匹配的id,这里不再做修改。
自动注入并不是很推荐,如果有兴趣可以看看这个挺全的自动注入,这里我贴上各种模式的介绍:

模式 作用
no 这是默认设置,意味着它没有使用自动装配模式。你应该显示的使用bean引用来连接
byName 由属性名自动装配。Spring 容器看到bean采用了自动装配byName模式(autowire="byName"),然后根据它的属性在Spring 容器中寻找与属性名相同bean进行关联
byType 由属性的数据类型自动装配。Spring容器看到bean采用了自动装配的byType模式(autowire="byType"),然后根据属性类型在Spring容器中寻找与属性类型相同bean进行关联。如果存在不止一个这样的bean,将抛出异常。
constructor 类似于 byType,但该类型适用于构造函数参数类型。如果在容器中没有一个构造函数参数类型的 bean,则一个致命错误将会发生
autodetect Spring首先尝试通过 constructor 使用自动装配来连接,如果它不执行,Spring 尝试通过 byType 来自动装配

List注入

最后说一种非常不常用的注入

  1. <property name="list">
  2. <!-- 同样可以注入列表,也不常用 -->
  3. <list>
  4. <value>159</value>
  5. <value>qqq</value>
  6. <value>www</value>
  7. <value>eeee</value>
  8. </list>
  9. </property>

最后

正在学习中,如果有什么错误,请留言告诉我,会第一时间回复并修改。

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注