spring IOC简单入门

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方式

项目结构图 下面是注入的实例 先一个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

运行结果如下

 

 

上面使用的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和两种实现形式

posted @ 2016-08-17 15:17  docyard  阅读(216)  评论(0编辑  收藏  举报