Spring装配bean

创建对象之间协作关系的行为通常称为装配(wiring),这也是DI的本质。Spring提供了三种主要的装配机制。

  • 在XML中进行显式配置。
  • 在Java中进行显式配置。
  • 隐式的bean发现机制和自动装配。

隐式的bean发现机制和自动装配

Spring从两个角度来实现自动化装配:

  • 组件扫描(component scanning):Spring会自动发现应用上下文中所创建的bean。
  • 自动装配(autowiring):Spring自动满足bean之间的依赖。

通过xml启动组件扫描(也可以通过java文件启动,我感觉不方便,不使用了):

  1. <context:component-scan base-package="com.zjf" />

组件扫描会扫描上面配置目录下和子目录下的响应注解,然后注册为bean。自动转配也是通过注解来实现的。

当一个 Bean 被自动检测到时,会根据那个扫描器的 BeanNameGenerator 策略生成它的 bean 名称。默认情况下,对于包含 value 属性的 @Component、 @Repository、 @Service 和 @Controller,会把value 取值作为 Bean 的名字。如果这个注解不包含 value 值或是其他被自定义过滤器发现的组件,默认 Bean 名称会是小写开头的类的简单名称。

spring常用的注解(暂时不说Spring mvc的注解):

  • @Component

泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注。

通过value属性指定bean的名字。如@Component(value="xx"),或者@Component("xx")。

@Repository、@Service 和 @Controller是继承自@Component的,在现在的版本上,他们的用法没有任何差别。只是标识不同的层。以后的版本可能会有差别。

  • @Autowired

默认按类型装配,如果我们想使用按名称装配,可以结合@Qualifier注解一起使用。

如下:@Autowired @Qualifier("personDaoBean") 存在多个实例配合使用

@Autowired 有一个required属性,定义是否必须,默认是true。

  • @Resource

默认按名称装配。

按照名称装配的方法:如果定义了name属性,那么按照name属性配置的名称去查找bean,如果没有定义name属性,那么是把属性名称首字母小写。

如果按照名称装配没有找到,会按类型装配,跟Autowired一样。

  • @Scope注解 作用域
  • @Lazy(true) 表示延迟初始化
  • @Service用于标注业务层组件、
  • @Controller用于标注控制层组件
  • @Repository用于标注数据访问组件,即DAO组件。
  • @PostConstruct用于指定初始化方法(用在方法上)
  • @PreDestory用于指定销毁方法(用在方法上)
  • @DependsOn:定义Bean初始化及销毁时的顺序
  • @Primary:自动装配时当出现多个Bean候选者时,被注解为@Primary的Bean将作为首选者,否则将抛出异常
  • @PostConstruct 初始化注解
  • @PreDestroy 摧毁注解 默认 单例 启动就加载
  • @Async异步方法调用
  • @Scope用于指定scope作用域的(用在类上)

这里的 scope 就是用来配置 spring bean 的作用域,它标识 bean 的作用域。

在spring2.0之前bean只有2种作用域即:singleton(单例)、non-singleton(也称 prototype), Spring2.0以后,增加了session、request、global session三种专用于Web应用程序上下文的Bean。

1、singleton 作用域

当一个bean的 作用域设置为singleton, 那么Spring IOC容器中只会存在一个共享的bean实例,并且所有对bean的请求,只要id与该bean定义相匹配,则只会返回bean的同一实例。换言之,当把 一个bean定义设置为singleton作用域时,Spring IOC容器只会创建该bean定义的唯一实例。这个单一实例会被存储到单例缓存(singleton cache)中,并且所有针对该bean的后续请求和引用都 将返回被缓存的对象实例,这里要注意的是singleton作用域和GOF设计模式中的单例是完全不同的,单例设计模式表示一个ClassLoader中 只有一个class存在,而这里的singleton则表示一个容器对应一个bean,也就是说当一个bean被标识为singleton时 候,spring的IOC容器中只会存在一个该bean。

配置实例: 

<bean id="role" class="spring.chapter2.maryGame.Role" scope="singleton"/>

或者

<bean id="role" class="spring.chapter2.maryGame.Role" singleton="true"/>

2、prototype

prototype作用域部署的bean,每一次请求(将其注入到另一个bean中,或者以程序的方式调用容器的 getBean()方法)都会产生一个新的bean实例,相当与一个new的操作,对于prototype作用域的bean,有一点非常重要,那就是Spring不能对一个prototype bean的整个生命周期负责,容器在初始化、配置、装饰或者是装配完一个prototype实例后,将它交给客户端,随后就对该prototype实例不闻不问了。不管何种作用域,容器都会调用所有对象的初始化生命周期回调方法,而对prototype而言,任何配置好的析构生命周期回调方法都将不会被调用。 清除prototype作用域的对象并释放任何prototype bean所持有的昂贵资源,都是客户端代码的职责。(让Spring容器释放被singleton作用域bean占用资源的一种可行方式是,通过使用 bean的后置处理器,该处理器持有要被清除的bean的引用。)

配置实例: 

<bean id="role" class="spring.chapter2.maryGame.Role" scope="prototype"/>

或者

<beanid="role" class="spring.chapter2.maryGame.Role" singleton="false"/>

3、request

request表示该针对每一次HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP request内有效,配置实例:

request、session、global session使用的时候首先要在初始化web的web.xml中做如下配置:

如果你使用的是Servlet 2.4及以上的web容器,那么你仅需要在web应用的XML声明文件web.xml中增加下述ContextListener即可: 

<web-app>

...

<listener>

<listener-class>org.springframework.web.context.request.RequestContextListener</listener-class>

</listener>

...

</web-app>

如果是Servlet2.4以前的web容器,那么你要使用一个javax.servlet.Filter的实现: 

<web-app>

..

<filter>

<filter-name>requestContextFilter</filter-name>

<filter-class>org.springframework.web.filter.RequestContextFilter</filter-class>

</filter>

<filter-mapping>

<filter-name>requestContextFilter</filter-name>

<url-pattern>/*</url-pattern>

</filter-mapping>

...

</web-app>

接着既可以配置bean的作用域了: 

<bean id="role" class="spring.chapter2.maryGame.Role" scope="request"/>

4、session

session作用域表示该针对每一次HTTP请求都会产生一个新的bean,同时该bean仅在当前HTTP session内有效,配置实例:

配置实例:

和request配置实例的前提一样,配置好web启动文件就可以如下配置: 

<bean id="role" class="spring.chapter2.maryGame.Role" scope="session"/>

5、global session

global session作用域类似于标准的HTTP Session作用域,不过它仅仅在基于portlet的web应用中才有意义。Portlet规范定义了全局Session的概念,它被所有构成某个 portlet web应用的各种不同的portlet所共享。在global session作用域中定义的bean被限定于全局portlet Session的生命周期范围内。如果你在web中使用global session作用域来标识bean,那么web会自动当成session类型来使用。

在xml中显示配置:

这种情况一般不能使用注解的情况下才会使用,不做深入学习,摘抄一篇文章备用。

文章地址:http://www.cnblogs.com/le576169235/p/6203129.html

(2)配置一个简单的bean

  <bean class="pojo.TestSpring" id="test"></bean>

注:上例中id属性如果没有指定,这个bean将会根据全限定类名命名,在上例中,将会是"pojo.TestSpring#0"。其中#0为一个计数形式,用来区分其他同类型的bean,若声明了

另外一个TestSpring,将会是"pojo.TestSpring#1"。

(3)借助构造器注入属性

constructor-arg节点

上例中constructor-arg节点的顺序即为构造函数中参数列表的参数顺序,一一对应,类型对应错误则会抛出异常

同时,该节点数目与要使用的构造函数的参数列表的参数个数必须一致。上例中注入的是对象类型,若要注入字面

良,讲ref属性改为value即可,如

此外,使用构造器注入属性除了使用constructor-arg节点外,还可以使用c命名空间,使用c命名空间可以减少配置文件的

冗长,但是constructor-arg节点能做到的有些事情,c命名空间无法做到。

c命名空间

1.要使用c命名空间,必须要在xml文件顶部声明其模式,如下图所示。

2.c命名空间格式

c:cdplay-ref="play"

c:c命名空间前缀 cdplay:构造器参数名 -ref:注入bean引用  ="play" :要注入的bean的ID

c命名空间也可根据参数顺序注入属性,即顺序索引,因为XML不支持数字作为属性的第一个字符,所以前数字前加上一个下划线

 

与constructor-arg参数一样,注入属性类型与数量,必须与使用的构造函数一致。

若要通过c命名空间注入字面量,则如下图所示

笔者所给出的样例类中并没有给出String,int等类型的属性,这里给出的字面量注入方式只作为样例师范,读者若要经行测试需要自己建立更优的样例类。

(4)通过属性的set方法注入属性

<property></property>节点

其中name为属性名,ref为要注入的bean的Id

与construstor-arg节点类似,若要注入字面量,将ref属性改为value

p命名空间

1.要使用p命名空间,需要在XML文件顶部配置其模式,如下图

2.p命名空间格式

p:cdplay-ref="play"

p:p命名空间前缀 cdplay:构造器参数名 -ref:注入bean引用  ="play" :要注入的bean的ID

 

p命名空间不能通过顺序索引注入属性

与c命名空间类似,若要注入字面量,将-ref去掉。

(5)集合类型属性的注入

   

如上图所示,给出了list,map,set集合类型的注入方式,此外可借助util命名空间创建集合bean

1.要使用util命名空间,需要在xml文件顶部配置其模式

2.util:list

util命名空间让集合能够像其他的bean一样被注入到其他bean中

3.util-命名空间中的元素

 元素

描述

<util:constant>

引用某个类型的Public static域,并将其暴露为bean

<util:list>

创建一个java.util.list类型的bean,其中包含值或引用

<util:map>

创建一个java.util.map类型的bean,其中包含值或引用

<util:properties>

创建一个java.util.Properties类型的bean

<util:property-path>

引用一个bean的属性(或内嵌属性),并将其暴露为bean

<util:set>

创建一个java.util.Set类型的bean,其中包含值或引用

在java中显示装备:

通过@Configuration 和@Bean 注解来实现,@Configuration把一个类作为一个IoC容器,它的某个方法头上如果注册了@Bean,就会作为这个Spring容器中的Bean。

很少使用,不再赘述。

原文地址:https://www.cnblogs.com/xiaolang8762400/p/7018709.html