spring(一):spring IoC的注入方式总结


前言:谈谈对springIOc的理解,两张图很好的阐述了springIoc容器的作用。

      

            传统应用程序示意图.jpg

       

         

            IoC容器后程序结构示意图.jpg

 


 

 

    springIoC容器注入方式有set注入,构造器注入,注解注入。

一:set方式注入

1.先准备几个bean类,一个普通的学生类,以及一个A类,A类组合了学生类对象。

Student.java

package com.cnblogs.bean; public class Student { // 学号 
    private String sNo; private String name; private int age; // 性别 
    private String sex; //年级
    private String grade; public Student() { super(); } public Student(String sNo, String name, int age, String sex, String grade) { super(); this.sNo = sNo; this.name = name; this.age = age; this.sex = sex; this.grade = grade; } 
  // set和get方法
  
  // toString方法()
}

 

A.java

package com.cnblogs.bean; public class A { private String desc; private Student stu; public A() { super(); // TODO Auto-generated constructor stub
 } public A(String desc, Student stu) { super(); this.desc = desc; this.stu = stu; } // set和get方法
  
  // toString方法()
}

 set.xml中

  <!-- 基于set方法的注入 -->
    <bean name="stu" class="com.cnblogs.bean.Student">
        <property name="sNo" value="1001"></property>
        <property name="name" value="jack"></property>
        <property name="age" value="12"></property>
        <property name="sex" value="male"></property>
        <property name="grade" value="三年级"></property>
    </bean>
    <!-- 注入引用类型 -->
    <bean name="A" class="com.cnblogs.bean.A">
        <property name="desc" value="A组合了Student类对象"></property>
        <!-- ref属性表示调用这个setStudent方法的时候要用的参数是名字为stu的对象 -->
        <property name="stu" ref="stu"></property>
    </bean>

1.<bean>标签中,name和id起标识这个对象的作用,id会帮我们检查给对象起的名字是否规范(名字不能重复,不能有空格,不能已数字开头),name不会检查这些东西。

class属性是一个类的全限定名,标识配置那个类。

2.springIoC默认是已单例模式管理对象,即通过相同的名字多次拿出来的对象一样,可以再<bean>标签中加属性 scope="prototype"代表非单例,

scope="Singleton"代表单例模式。

3.可以给某一个对象加别名,在</bean>后面加一条<alias name="stu" alias="s1"/>。可以通过stu拿对象,也可以通过s1 拿对象。

4.当类中的一个成员变量为另一个类的对象时,在<property>子标签中可以通过ref引入,ref的值为一个<bean>标签的name或id值。

5.同一个类可以配置多个对象,但是标识多个对象的id或name值要不同。

注意:set方式底层会用到一个bean类的set方法,如果bean类的成员变量没有set方法却采用了set方式注入会报错。

测试方法:

  /** * 基于set方法的注入 */ @Test public void set(){ try{ String[] path = {"com/cnblogs/ioc/set/set.xml"}; ApplicationContext container = new ClassPathXmlApplicationContext(path); Student stu = (Student)container.getBean("stu"); System.out.println(stu); Object a = container.getBean("A"); System.out.println(a); } catch(Exception e) { e.printStackTrace(); } }

 

 结果:

 Student [sNo=1001, name=jack, age=12, sex=male, grade=三年级]
A [desc=A组合了Student类对象, stu=Student [sNo=1001, name=jack, age=12, sex=male, grade=三年级]]

 

2.set方式注入之集合的注入

(1)准备一个B类,里面包括了各种集合:list,set,map,property

B.java

package com.cnblogs.bean; import java.util.List; import java.util.Map; import java.util.Properties; import java.util.Set; public class B { // list集合
    private List<String> list; // set集合
    private Set<String> set; // map
    private Map<String,String> map; // prop
    private Properties prop;  // set和get方法 }

collection.xml

   <!-- 集合的注入 -->
    <bean name="coll" class="com.cnblogs.bean.B">
        <!-- list集合 -->
        <property name="list">
            <list>
                <value>jack</value>
                <value>tom</value>
                <value>tina</value>
            </list>
        </property>
        <!-- set集合 -->
        <property name="set">
            <set>
                <value>1001</value>
                <value>1002</value>
                <value>1003</value>
            </set>
        </property>
        <!-- map集合 -->
        <property name="map">
            <map>
                <entry key="name" value="zhangsan"></entry>
                <entry key="age" value="12"></entry>
                <entry key="gender" value="male"></entry>
            </map>
        </property>
        <!-- properties集合注入 -->
        <property name="prop">
            <props>
                <prop key="driver">com.mysql.jdbc.Driver</prop>
                <prop key="username">study</prop>
            </props>
        </property>
    </bean>

 

测试方法:和上面那个一样,就是xml的文件路径改一下。

 

(2):如果一个list集合的泛型是一个类,也可以注入。

HobbyGroup.java

package com.cnblogs.bean; import java.util.List; // 兴趣小组
public class HobbyGroup { private Long id; private String name; private List<Student> stu; public HobbyGroup() { super(); // TODO Auto-generated constructor stub
 } public HobbyGroup(Long id, String name, List<Student> stu) { super(); this.id = id; this.name = name; this.stu = stu; }
// set和get方法
  
  // toString方法()
}

collection.xml

  <bean name="hg" class="com.cnblogs.bean.HobbyGroup">
        <property name="id" value="1"></property>
        <property name="name" value="羽毛球兴趣小组"></property>
        <property name="stu">
            <list>
                <bean name="stu1" class="com.cnblogs.bean.Student" >
                    <property name="sNo" value="1001"></property>
                    <property name="name" value="jack"></property>
                    <property name="age" value="12"></property>
                    <property name="sex" value="male"></property>
                    <property name="grade" value="三年级"></property>
                </bean>
                <bean name="stu2" class="com.cnblogs.bean.Student">
                    <property name="sNo" value="1002"></property>
                    <property name="name" value="tom"></property>
                    <property name="age" value="13"></property>
                    <property name="sex" value="male"></property>
                    <property name="grade" value="三年级"></property>
                </bean>
                <bean name="stu3" class="com.cnblogs.bean.Student">
                    <property name="sNo" value="1003"></property>
                    <property name="name" value="tina"></property>
                    <property name="age" value="11"></property>
                    <property name="sex" value="female"></property>
                    <property name="grade" value="三年级"></property>
                </bean>
            </list>
        </property>
    </bean>

 

3.set注入之自动注入

1.自动注入一般针对一个类中组合了另一类的对象。

2.自动注入有byName注入和byType注入

3.byName注入:spring容器会到当前的类中找property的名字,然后再根据这个名字去spring容器中找有没有和这个property名字相同的对象,有的话,

 就把这个对象当做参数放到setXxxx这个方法里面注入进来。(找到多个不会报错)

4.byType注入:spring容器会根据set方法的参数类型去容器中找相匹配的对象,找到就注入,没找到就算了。如果找到多个会报错。

5.可以在<beans>标签中加 default-autowire="byType",则下面的标签会根据byType方式自动注入。

6.可以在<bean>标签中autowire=" "指定注入方式,该方式会屏蔽 default-autowire=" "。

示例:

同样用到com.cnblogs.bean.A类和com.cnblogs.bean.Student类。

autowired.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" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.3.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd" default-autowire="byName"
>

    <bean name="stu" class="com.cnblogs.bean.Student">
        <property name="sNo" value="1001"></property>
        <property name="name" value="jack"></property>
        <property name="age" value="12"></property>
        <property name="sex" value="male"></property>
        <property name="grade" value="三年级"></property>
    </bean>
<!-- byName --> <bean name="A1" class="com.cnblogs.bean.A"> <property name="desc" value="A组合了Student类对象,byName"></property> </bean> <!-- byType --> <bean name="A2" class="com.cnblogs.bean.A" autowire="byType"> <property name="desc" value="A组合了Student类对象,byType"></property> </bean> </beans>

 

测试方法:

  @Test public void autowired(){ try{ String[] path = {"com/cnblogs/ioc/autowired/autowired.xml"}; ApplicationContext container = new ClassPathXmlApplicationContext(path); // byName
            A a1 = (A) container.getBean("A1"); System.out.println(a1); // byType
            A a2 = (A) container.getBean("A2"); System.out.println(a2); } catch(Exception e) { e.printStackTrace(); } }

 

 结果:

A [desc=A组合了Student类对象,byName, stu=Student [sNo=1001, name=jack, age=12, sex=male, grade=三年级]]
A [desc=A组合了Student类对象,byType, stu=Student [sNo=1001, name=jack, age=12, sex=male, grade=三年级]]


 

二:构造器注入

构造器有分两种注入方式,一个根据参数类型注入,一个根据下标注入。

1.根据参数类型注入

  同样用到com.cnblogs.bean.Student类。

  constructor.xml

   <!-- 根据参数类型 -->
    <bean name="stu1" class="com.cnblogs.bean.Student">
        <constructor-arg type="String" value="1001"></constructor-arg>
        <constructor-arg type="String" value="jack"></constructor-arg>
        <constructor-arg type="int" value="12"></constructor-arg>
        <constructor-arg type="String" value="male"></constructor-arg>
        <constructor-arg type="String" value="三年级"></constructor-arg>
    </bean>

 

2.根据参数下标注入

  <!-- 根据下标 -->
    <bean name="stu2" class="com.cnblogs.bean.Student">
        <constructor-arg index="0" value="1002"></constructor-arg>
        <constructor-arg index="1" value="tina"></constructor-arg>
        <constructor-arg index="2" value="12"></constructor-arg>
        <constructor-arg index="3" value="female"></constructor-arg>
        <constructor-arg index="4" value="三年级"></constructor-arg>
    </bean>

 

 


 

三、注解注入

首先需要在xml文件中指定使用注解注入的包,springIoC容器读取这个文件的时候就会知道。

anatation.xml:

<context:component-scan base-package="com.cnblogs.ioc.anatation" />   

 

com.cnblogs.ioc.annotation.Office.java

package com.cnblogs.ioc.anatation; import org.springframework.stereotype.Component; @Component public class Office { private String num = "001"; public Office(){ } public Office(String num) { this.num = num; } public String getNum() { return num; } public void setNum(String num) { this.num = num; } }

 

com.cnblogs.ioc.annotation.Car.java

package com.cnblogs.ioc.anatation;
import org.springframework.stereotype.Component;
@Component
public class Car { private double price; private String name; public Car(){ } public Car(double price, String name) { this.price = price; this.name = name; } public double getPrice() { return price; } public void setPrice(double price) { this.price = price; } public String getName() { return name; } public void
setName(String name) { this.name = name; } }

 

com.cnblogs.ioc.annotation.Boss.java

package com.cnblogs.ioc.anatation; import javax.annotation.PostConstruct; import javax.annotation.PreDestroy; import javax.annotation.Resource; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Scope; import org.springframework.stereotype.Component; @Component @Scope("prototype") public class Boss { private String name; @Autowired private Car car; @Resource private Office office; public Boss(){ } public Boss(String name, Car car, Office office) { this.name = name; this.car = car; this.office = office; } public Boss( Car car, Office office) { this.car = car; this.office = office; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Car getCar() { return car; } public void setCar(Car car) { this.car = car; } public Office getOffice() { return office; } public void setOffice(Office office) { this.office = office; } @PostConstruct public void init(){ System.out.println("初始化.."); } @PreDestroy public void destory(){ System.out.println("销毁..."); } 
}

 测试方法:

  /** * 注解注入 */ @Test public void anatation(){ try{ String[] path = {"com/cnblogs/ioc/anatation/anatation.xml"}; ApplicationContext container = new ClassPathXmlApplicationContext(path); Boss boss = (Boss) container.getBean("boss"); System.out.println(boss.getName()); System.out.println(boss.getCar()); System.out.println(boss.getOffice()); System.out.println(boss); } catch(Exception e) { e.printStackTrace(); } }

 

 结果

 初始化..
null
com.cnblogs.ioc.anatation.Car@50de0926
com.cnblogs.ioc.anatation.Office@2473b9ce
com.cnblogs.ioc.anatation.Boss@60438a68

 

 几个注解的作用:

@Autowired 1) @Autowired使用后需要在xml文件加入以下配置才能生效: <context:annotation-config/> 2) @Autowired注解可以写在成员变量、setter方法、构造器函数上面 3) @Autowired默认按照byType匹配的方式进行注入,如果没有一个bean的类型是匹配的则会抛异常,如果有多个bean的类型都匹配成功了, 那么再按byName方式进行选择 4) @Autowired如果最终匹配不成功(注意一定是一个都没有找到的情况)则会抛出异常,但是如果设置为 @Autowired(required=false), 则最终匹配不成功没有不会抛出异常。 5) @Autowired可以结合@Qualifier("beanName")来使用,则可以达到byName的效果 @Resource 1) @Resource使用后需要在xml文件加入以下配置才能生效:<context:annotation-config/> 2) @Resource的作用和@Autowired差不多,只不过 @Resource是默认先用byName,如果找不到合适的就再用byType来注入 3) @Resource有俩个属性,name和type,使用name属性则表示要byName匹配,使用type属性则表示要byType匹配 @PostConstruct和@PreDestroy 1) 标注了@PostConstruct注解的方法将在类实例化后调用 2) 标注了@PreDestroy注解的方法将在类销毁之前调用 @Component 1) @Component注解可以直接定义bean,而无需在xml定义。但是若两种定义同时存在,xml中的定义会覆盖类中注解的Bean定义 2) @Component注解直接写在类上面即可 3) @Component有一个可选的参数,用于指定bean的名称 @Component("boss") public class Boss{} 4) @Component如果不指定参数,则bean的名称为当前类的类名小写 //和上面例子的相关相同 @Component public class Boss{} 5) @Component使用之后需要在xml文件配置一个标签 <context:component-scan/> 6) <context:component-scan base-package="com.briup.ioc.annotation" /> 表示spring检查指定包下的java类,看它们是否使用了 @Component注解 7) @Component定义的bean默认情况下都是单例模式的,如果要让这个bean变为非单例,可以再结合这个@Scope注解来达到目标@Scope("prototype") @Component是Spring中所有bean组件的通用形式, @Repository @Service @Controller 则是 @Component的细化,用来表示更具体的用例, 分别对应了持久化层、服务层和表现层。但是至少到现在为止这个四种注解的实质区别很小(甚至几乎没有),都是把当前类注册为spring容器中 的一个bean 注意: 1.component-scan标签默认情况下自动扫描指定路径下的包(含所有子包) 2.component-scan标签将带有@Component @Repository @Service @Controller注解的类自动注册到spring容器中 3.component-scan标签对标记了@Required @Autowired @PostConstruct @PreDestroy @Resource @WebServiceRef @EJB @PersistenceContext @PersistenceUnit等注解的类进行对应的操作,使注解生效 4.component-scan标签包含了annotation-config标签的作用

 

 

 

 

 


免责声明!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系本站邮箱yoyou2525@163.com删除。



 
粤ICP备18138465号  © 2018-2025 CODEPRJ.COM