spring IOC简单入门

时间:2024-08-04 09:33:56

spring的核心是ioc和aop

先介绍一下IOC(inverse of control控制反转)又叫DI(Dependency injection依赖注入)

个人理解为把对象的控制权由类转移到配置文件中   把类所需要的对象通过配置文件注入到类中  可以表述不太准确

我们可以模拟spring 的BeanFactory和ClassPathXmlApplicationContext 但是这不重要

package com.ouc.wkp.spring;

public interface BeanFactory {
public Object getBean(String id);
}

BeanFactory.java

package com.ouc.wkp.spring;

import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.List;
import java.util.Map; import org.jdom2.Document;
import org.jdom2.Element;
import org.jdom2.input.SAXBuilder; public class ClassPathXmlApplicationContext implements BeanFactory{ private Map<String, Object> beans=new HashMap<String,Object>(); public ClassPathXmlApplicationContext() throws Exception{
SAXBuilder sb = new SAXBuilder(); Document doc = sb.build("src\\com\\ouc\\wkp\\test\\beans.xml"); // 构造文件对象
Element root = doc.getRootElement(); // 获取根元素HD
List<?> list = root.getChildren("bean");// 取名字为bean的所有元素
for (int i = 0; i < list.size(); i++) {
Element element = (Element) list.get(i);
String id = element.getAttributeValue("id");
String clazz = element.getAttributeValue("class");
Object o=Class.forName(clazz).newInstance();
System.out.println(id);
System.out.println(clazz);
beans.put(id,o); for(Element propertyElement:(List<Element>)element.getChildren("property")){
String name=propertyElement.getAttributeValue("name");
String bean=propertyElement.getAttributeValue("bean"); //u
Object beanObject=beans.get(bean); //UserDAOImpl instance
String methodName="set" +name.substring(0,1).toUpperCase()+name.substring(1);
System.out.println("method name="+methodName); Method m=o.getClass().getMethod(methodName, beanObject.getClass().getInterfaces()[0]);//setUserDAO(UserDAO)
m.invoke(o, beanObject);
}
}
} @Override
public Object getBean(String id) {
return beans.get(id);
} }

ClassPathXmlApplicationContext.java

代码中的路径需要更改

介绍注入的两种方式xml和annotation,先介绍xml方式

spring  IOC简单入门

项目结构图 下面是注入的实例 先一个User实体类

package com.ouc.wkp.model;

public class User {
private String username;
private String password; public String getUsername() {
return username;
} public void setUsername(String username) {
this.username = username;
} public String getPassword() {
return password;
} public void setPassword(String password) {
this.password = password;
}
}

User.java

一个接口UserDAO

package com.ouc.wkp.dao;

import com.ouc.wkp.model.User;

public interface UserDAO {
public void save(User user);
}

UserDAO.java

接口实现UserDAOImpl

package com.ouc.wkp.dao.impl;

import java.util.List;
import java.util.Map;
import java.util.Set; import com.ouc.wkp.dao.UserDAO;
import com.ouc.wkp.model.User; public class UserDAOImpl implements UserDAO { public void save(User user) {
// Hibernate
// JDBC
// XML
// NetWork
System.out.println("user saved!");
} private Set<String> sets;
private List<String> lists;
private Map<String, String> maps; public Set<String> getSets() {
return sets;
} public void setSets(Set<String> sets) {
this.sets = sets;
} public List<String> getLists() {
return lists;
} public void setLists(List<String> lists) {
this.lists = lists;
} public Map<String, String> getMaps() {
return maps;
} public void setMaps(Map<String, String> maps) {
this.maps = maps;
} public UserDAOImpl() {
} @Override
public String toString() {
return "sets size:" + sets.size() + "| lists size:" + lists.size()
+ "| maps size:" + maps.size();
} }

UserDAOImpl.java

上面代码里面的list set map 和 toString方法是测试通过配置文件给对象里面的属性赋值

然后是UserService

package com.ouc.wkp.service;

import com.ouc.wkp.dao.UserDAO;
import com.ouc.wkp.model.User; public class UserService {
private UserDAO userDAO;
private UserService(UserDAO userDAO) {
this.userDAO = userDAO;
}
public void init(){
System.out.println("init");
}
public void destroy(){
System.out.println("destroy");
}
public void add(User user) {
userDAO.save(user);
}
public UserDAO getUserDAO() {
return userDAO;
}
public void setUserDAO(UserDAO userDAO) {
this.userDAO = userDAO;
}
}

UserService.java

然后是两个配置文件,命名比较随意

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd">
<!-- default-autowire= -->
<!-- 在spring中,singleton属性默认是true,只有设定为false,则每次指定别名取得的Bean时都会产生一个新的实例 -->
<bean id="u" class="com.ouc.wkp.dao.impl.UserDAOImpl" />
<!-- 自动装配 autowire -->
<!-- scope 生命范围 singleton单例 prototype每次创建新的对象 -->
<!-- init-method="" destory-method=""-->
<bean id="userservice" class="com.ouc.wkp.service.UserService" scope="prototype" init-method="init" destroy-method="destroy">
<!-- 指set方法 注入 -->
<!-- <property name="userDAO" ref="u" /> -->
<constructor-arg>
<ref bean="u" />
</constructor-arg>
</bean>
</beans>

beans.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-2.5.xsd"> <bean id="userDAO" class="com.ouc.wkp.dao.impl.UserDAOImpl">
<property name="sets">
<set>
<value>1</value>
<value>2</value>
</set>
</property>
<property name="lists">
<list>
<value>1</value>
<value>2</value>
<value>3</value>
</list>
</property>
<property name="maps">
<map>
<entry key="1" value="1"></entry>
<entry key="2" value="2"></entry>
<entry key="3" value="3"></entry>
<entry key="4" value="4"></entry>
</map>
</property>
</bean> <bean id="userService" class="com.ouc.wkp.service.UserService">
<!--
<property name="userDAO">
<ref bean="userDAO"/>
</property>
-->
<constructor-arg>
<ref bean="userDAO"/>
</constructor-arg>
</bean> </beans>

beans1.xml

我们看一下配置文件

<beans></beans>是最外层标签

第一层子元素是<bean></bean>标签 代表一个类实例化时的对象

属性id用于唯一标识这个对象。

class用于指定这个类文件的位置,输入完毕后按ctrl,如果生成可以跳转的链接说明输入正确。

scope代表生命范围,默认是singleton即单例模式,如果设置为prototype则每次实例化都生成一个新的对象。

init-method="init"  destroy-method="destroy"标识对象初始化和销毁时分别调用init() 和destroy()方法。

可以设置autowire来进行自动装配,但是有时容易产生混乱,也可以在<beans>标签里面加上default-autowire。

然后是重点

<bean id="userservice" class="com.ouc.wkp.service.UserService" scope="prototype" init-method="init" destroy-method="destroy">
<!-- 指set方法 注入 -->
<!-- <property name="userDAO" ref="u" /> -->
<constructor-arg>
<ref bean="u" />
</constructor-arg>
</bean>

看UserService.java这个类里面有个

private UserDAO userDAO;

然后我们为他写上get set方法。这样就可以通过配置文件注入这个对象。

我们注意到注入的bean ref=“u” 这个u和和上面声明的userDAO的id相同。

注入有3种方法 第一种是上面注释的set方法注入 第二种是上面的构造方法注入

因为我们在UserService里面有构造方法

private UserService(UserDAO userDAO) {
this.userDAO = userDAO;
}

所以我们必须要采用构造方法注入,如何注释掉构造方法就可以采用第一种。

第三种接口注入,没有去了解。

最后是一个junit测试类

package com.ouc.wkp.test;

import static org.junit.Assert.*;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; import com.ouc.wkp.dao.UserDAO;
import com.ouc.wkp.model.User;
import com.ouc.wkp.service.UserService;
//import com.ouc.wkp.spring.BeanFactory;
//import com.ouc.wkp.spring.ClassPathXmlApplicationContext; public class UserServiceTest { @Test
public void testAdd() throws Exception{
ClassPathXmlApplicationContext ctx1=new ClassPathXmlApplicationContext("beans.xml"); UserService service=(UserService)ctx1.getBean("userservice");
UserService service2=(UserService)ctx1.getBean("userservice"); System.out.println(service==service2); User user=new User();
user.setUsername("heihei");
user.setPassword("1234");
service.add(user); ApplicationContext ctx2=new ClassPathXmlApplicationContext("beans1.xml"); UserDAO userDAO=(UserDAO)ctx2.getBean("userDAO"); System.out.println(userDAO);
} }

UserServiceTest.java

运行结果如下

spring  IOC简单入门

上面使用的xml方式,下面介绍annotation方式,方便了很多。

首先需要在配置文件里面加上这些

xmlns:context="http://www.springframework.org/schema/context" 
 http://www.springframework.org/schema/context 
 http://www.springframework.org/schema/context/spring-context-3.1.xsd"> 
 <context:annotation-config />

使用这种方式,我们就不需要在xml中声明bean。上面xml中的两个bean标签可以删去。

两个类作了修改

package com.ouc.wkp.dao.impl;

import org.springframework.stereotype.Component;

import com.ouc.wkp.dao.UserDAO;
import com.ouc.wkp.model.User; @Component("u")
public class UserDAOImpl implements UserDAO { @Override
public void save(User user) {
// Hibernate
// JDBC
// XML
// NetWork
System.out.println("user saved!");
// throw new RuntimeException();
} @Override
public void delete() {
// TODO Auto-generated method stub } }

UserDAOImpl

在类的开头写上@Component("u")

package com.ouc.wkp.service;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import javax.annotation.Resource; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Component; import com.ouc.wkp.dao.UserDAO;
import com.ouc.wkp.model.User; @Component("userService")
public class UserService { private UserDAO userDAO; @PostConstruct
public void init() {
System.out.println("init");
} public void add(User user) {
userDAO.save(user);
} public UserDAO getUserDAO() {
return userDAO;
} // @Autowired @Qualifier("u")
@Resource(name = "u")
public void setUserDAO(UserDAO userDAO) {
this.userDAO = userDAO;
} @PreDestroy
public void destroy() {
System.out.println("destroy");
}
}

UserService.java

在类的开头写上@Component("userService")

在初始化方法前面写上@PostConstruct

在销毁方法前面写上@PreDestroy

在setUserDAO前面写上@Resource(name = "u")

注意这个name=“u”  和@Component("u")的对应关系

下一篇简单介绍aop和两种实现形式