Spring Bean 配置三种方式:
- XML显式配置
- Java中显式配置
- 隐式Bean发现机制和自动装配。
隐式Bean发现机制
- @Compone(@Named 基本上等价) 声明Bean
- @Compone("aaa")可以制定Bean id,默认id为Bean的首字母小写.
- @Comfiguration 声明一个类为Java类,并用@ComponentScan开启自动发现机制,或在XML中<context:componect-scan base-package="包名">
- @ComponentScan 可以指定多个包名
@ComponentScan(basePackages = {"soundsfs","asdfs"})
或者指定一个类,这Spring会去搜索该类所在的包@ComponentScan(basePackageClasses = CompactDisc.class)
,在每个包下声明一个mark interface用于发现Bean。 - 对需要注入的类进行@Autowired (@Inject 基本上等价)进行自动装配,这样Spring就可以进行搜索,从而设定符合要求的类进行注入
- 如果自动装配的类不存在将会报错。
org.springframework.beans.factory.NoSuchBeanDefinitionException
- 如果多个符合标准也会报错
org.springframework.beans.factory.NoUniqueBeanDefinitionException
Java显式配置
- 声明该Java类为配置类。@Configuration 表明该类为配置类。
- 在一个可以创建所需要类型实例的方法上@Bean 表面返回的该对象是需要注册到Spring上下文中的Bean。和XML中的<Bean> 相同。
- @Bean(name="bean自定义Id")
@Bean
public CDPlayer clPlayer(){return new CDPlayer();}
- Spring会对这样的方法调用进行拦截,并不是每一次都是完全的调用,可以直接放回该方法的的实例.Spring中Bean默认是单例的。
- Spring会对方法的参数进行自动装配。
@Bean
public CDPlayer clPlayer(A a){return new B(a);}
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.xsd">
<!--声明一个Bean 默认id=soundsystem.SgtPeppers#0-->
<!--id 指定Bean的ID,调用SgtPeppers的默认构造器-->
<bean class="soundsystem.SgtPeppers" id="compactDisc"/>
</beans>
- 构造器注入
<constructor-arg>或者Spring3.0 引入 c-命名空间 进行注入,但是前者功能更强大。
<bean class="soundsystem.SgtPeppers" id="sgtPeppers">
<constructor-arg ref="sgtPeppers" name="title"/>
</bean>
<bean class="soundsystem.SgtPeppers" id="sgtPeppers"
c:title-ref="ad"
c:_0-ref="aa"
/>
<!--title 是构造方法的参数名字,可以用index _0 表示第一个,-ref 表示后面是个Bean不是字符串,
_参数名="" 表示字符常量 _index 同理-->
- 属性注入
`<property name="asdf" ref="adsf"/> p:属性名-ref="asdf 用法与构造器注入完全一样.
导入配置与混合配置
- @Import 在配置类上,导入别的配置类
@Import(Config.class)
,创建一个专门用于整合的配置类 @Import({Config1.class,Config2.class}) - @ImportResource("classpath:cd-config.xml")引入xml配置文件
- xml引入配置文件
<import resource="spring-config.xml"/>
引入java配置文件,<bean class="Config.class">
Spring Profile 多环境切换配置
- 在@Profile 在配置类上或者是Bean上标识@Profile("dev") @Profile("prod") @Profile("环境") ,XML中Beans 的profile属性进行指定。只有当前设置的Profile与Bean的Profile相同时,返回同一个类型的Bean时,该Bean才会被实例化。没有Profile标识是正常实例化。
- 设置Profile
spring.profile.active spring.profile.default
,当active为设置时,会读取default的值,如果都没有设置,只会实例化没有profile标识的bean。 - active和default的值设置
- DispatcherServlet的初始化参数。
<context-param>
<param-name>spring.profile.default</param-name>
<param-value>dev</param-value>
</context-param>
<servlet>
......
.. ...
<init-param>
<param-name>spring.profile.default</param-name>
<param-value>dev</param-value>
</init-param>
</servlet>
- Web应用的上下文参数。
- JNDOI条目
- 环境变量
- jvm系统属性
- 测试类上,@ActiveProfiles注解
条件化Bean
@Conditional(实现了Condition接口的类.class),只有当matches返回true时,才会创建该Conditional标记的类。@Profile就是基于Conditional实现的。
自动装配的歧义性
- @Primary 设置该Bean为首选项 primary=true
- 使用限定符,通过一层一层的限定,将结果限制在一个@Qualifier。
@Qualifier(id) 指定id。
@Quelifier 对该组件进行描述。 在Autowired上设置Quelifier选择符合条件的bean。但是java不能在同一个上重复使用。so,自定义一个注解来完成多级限定
@Target({ElementType.FIELD, ElementType.METHOD, ElementType.PARAMETER, ElementType.TYPE, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Qualifier
public @interface Cold {
}
通过这样声明多个注解,多bean就声明了多个关键字,在使用是,通过这些关键字,进行过滤。
Bean的作用域
默认情况下,Bean的都是单例的,不管注入多少次都是同一个实例。
单例(Singleton) 原型(Prototype)会话(Session) 请求(Request)
声明组件是@scope设置作用域。例如
@Component
@Scope(value=WebApplicationContext.SCOPE_SESSION,proxyMode=ScopedProxyMode.INTERFACES)
public ShoppingCart cart(){}
这是代理模型,是因为当没有会话的时候,ShoppingCart是不能注入到别的Bean中的,而别的类又已经需要注入这个ShoppingCart对象,所以注入一个代理类,代理与Cart实现一样的接口,通过代理进行调用真正的Cart对象,同时别的类也可以正常实例化。当需要代理的类不是接口,而是一个类的时候 需要使用的proxyMode是ScopedProxy.TARGET_CLASS.
XML中设置作用域 scope="session"
设置代理 <aop:scoped-proxy> 默认等价与Target_Class, proxy-target-class="false"实现接口代理。
网友评论