三种主要装配方式:
- 自动化配置
- 基于Java的显示配置
- 基于XML的显示配置
对比了三种配置,觉得工作中应尽可能使用自动化配置,避免显示配置所带来的维护成本。 如果确实需要显示配置,优先Java配置,比XML配置更强大,类型更安全且易于重构。
profile的激活
Spring在确定哪个profile处于激活状态时,需要依赖两个独立属性:
spring.profiles.active和spring.profiles.default。
有多重方法设置这两种属性:
- DispatcherServlet的参数
- Web应用的上下文参数
- JNDI条目
- 环境变量
- JVM系统属性
- 集成测试类中用@ActiveProfile设置
从Spring 4 开始,@Profile注解进行了重构,使其基于@Conditional和Condition实现。实现见“Spring bean 相关Annotation收集整理”
处理自动装配的歧义性
如果不仅有一个bean能够匹配结果,歧义性会阻碍Spring自动装配属性、构造器参数或方法参数。比如用@Autowired
标注了某个X的set方法,而X是一个有3个实现类的接口,且这三个类均使用了@Component
注解。此时Spring会抛出NoUniqueBeanDefinitionException。
解决办法:
1)标示首选的Bean,通过@Primary
。(同一接口下只能有一个Primary bean)
2)使用限定符
-
@Qualifier("classname")
,classname可以为自定义的限定符,而不是依赖于将beanID作为限定符。 - 使用自定义的限定符注解:
@Target({ElementType.CONSTRUCTOR, ElementType.FIELD,
ElmentType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Qualifier
public @interface YourAnnotation{ }
然后,
@Autowired
@Qualifier("自定义限定符名")
@YourAnnotation
public void setAA(AA aa){
this.aa = aa;
}
Bean的作用域
Spring定义了多种作用域,可以基于这些作用域创建bean,包括:
- 单例(Singleton):在整个应用中,只创建bean的一个实例。
- 原型(Prototype):每次注入或通过Spring应用上下文获取时,都会创建一个新的bean实例。
- 会话(Session):在Web应用中,为每个会话创建一个bean实例。
- 请求(Rquest):在Web应用中,为每个请求创建一个bean实例。
单例为默认作用域,但对于易变的类型不合适。如果选择其他的作用域,要使用@Scope
注解,可以与@Component
或@Bean
一起使用。
在运行时值注入
有时,我们可能会希望避免硬编码值,并让这些值在运行时再确定。Spring有两种在运行时求值的方式:
- 属性占位符(Property placeholder)
- Spring表达式语言(SpEL)
注入外部的值
使用外部属性来装配一个bean:
@Configuration
@PropertySource("classpath:/路径/app.properties") //声明属性源
public class ExpressiveConfig{
@Autowired
Environment env;
@Bean
public XX x(){
return new XX(
env.getProperty("x.title"),
env.getProperty("x.artist"));
}
}
在本例中,@PropertySource引用了类路径中名为app.properties
的文件。大致如下所示:
x.title = xxxxx
x.artist = xxxax
这个属性的文件会加载到Spring的Environment中,稍后可以从这里检索属性。同时,在x()中,会创建一个新的XX,他的构造器参数是从属性文件中获取的,而这时通过调用getProperty()实现的。
解析属性占位符
Spring支持将属性定义到外部的属性文件,并使用占位符值将其插入到Spring bean中。在Spring装配中,占位符的形式为使用${...}
包装属性名称。
使用SpEL进行装配
SpEL(Spring Expression Languae)是Spring3引入的一种强大简洁的将值装配到bean属性和构造器参数中的方式。
SpELl特性:
- 使用bean的id引用bean;
- 调用方法和访问对象的属性;
- 对值进行算术、关系和逻辑运算;
- 正则表达式匹配;
- 集合操作。
SpEL需要放入“#{...}”之中。
SpEL表达式可以引用其他的bean或其他bean的属性:
#{id.attributename}
与@Value结合:
public ClassName{
@Value("#{systemProperties['id.attributename']}") String attributename,
@Value("#{systemProperties['id.anotherattributename']}") String anotherattributename){
this.attributename= attributename;
this.anotherattributename= anotherattributename;
}
还有很多其他功能,有机会整理一篇具体功能再发。