Spring提供了一个org.springframework.beans.factory.FactoryBean工厂类接口,用户可以通过实现该接口定制实例化Bean的逻辑。
从Spring3.0开始,FactoryBean开始支持泛型,即接口声明改为FactoryBean<T>的形式,在该接口中共定义了以下3种方法:
T getObject():返回由FactoryBean创建的Bean的实例,若isSingleton()返回的是true,则该实例会放到Spring容器的单实例缓存池中
boolean isSingleton():确定由FactoryBean创建的Bean的作用域是singleton还是prototype
Class<T>getObjectType():返回FactoryBean创建Bean的类型
Bean的衍型注解:
@Component:用于对Bean实现类进行标注
@Repository:用于对DAO实现类进行标注
@Service:用于对Service实现类进行标注
@Controller:用于对Controller实现类进行标注
Spring提供了一个context命名空间,它提供了通过扫描类包以应用注解定义Bean的方式
<beans ...
xmlns:context="http://www.springframework.org/schema/context"
...
xsi:schemaLocation="http://www.springframework.org/schema/context/spring-context-3.0.xsd"
...
>
<context:component-scan base-package="xxx]xx.x">
声明context命名空间后,即可通过context命名空间的component-scan的base-package属性指定一个需要扫描的基类包,Spring容器将会扫描这个基类包里的所有类,并从类的注解信息中获取Bean的定义信息。若希望仅扫描特定的类而非基包下所有的类,那么可以使用resource-pattern属性过滤出特定的类
<context:component-scan base-package="xxx.xx.x" resource-pattern="xxx/*.class">
<context:include-filter>表示要包含的目标类,而<context:exclude-filter>表示要抛出在外的目标类。一个<context:component-scan>下可以拥有若干个<context:exclude-filter>和<context:include-filter>元素,这两个元素均支持多种类型的过滤表达式。
过滤表达式:
annotation 所有标注了XxxAnnotation的类。该类型采用目标累是否标注了某个注解进行过滤。
assignable 所有继承或扩展了XxxService的类。该类型采用目标类是否继承或扩展某个特定类进行过滤。
aspectj 所有类名以Service结束的类及继承或扩展它们的类。
regex 所有目录类包下的类。该类型采用正则表达式根据目录类的类名进行过滤
custom 采用XxxTypeFile通过代码的方式根据过滤原则。该类必须实现org.springframework.core.type.TypeFilter
Spring通过@Autowired注解实现Bean的依赖注入。@Autowired默认按类型匹配的方式,在容器查找匹配的bean,当且仅当仅有一个匹配的Bean时,Spring将其注入到@Autowired标注变量中。如果容器中没哟一个和标注变量类型匹配的Bean,Spring容器启动时将报NoSuchBeanDefinitionException的异常。若希望Spring即使找不到匹配的Bean完成注入也不要抛出异常,则可使用@Autowired(required=false)进行标注。默认情况下,required的属性值为true。若容器中有一个以上匹配的Bean时,则可以通过@Qualifier注解限定Bean的名称
@Autowired
@Qualifier("userDao")
private UserDao userDao;//若容器中存在userDao和otherUserDao
@Autowired还可以对成员变量及方法的入参进行标注,也可对入参标注@Qualifier以指定注入Bean的名称。
一般情况下,在Spring容器中大部分的Bean都是单实例的,所以我们一般都无须通过@Repository、@Service等注解的value属性为Bean指定名称,也无须使用@Qualifier按名称进行注入。
若Spring发现变量是一个集合类,则它会将容器中匹配集合元素类型的所有Bean都注入进来。
Spring还支持JSR250中定义的@Resource和JSR-330中定义的@Inject注解,这两个注解都是对类变量及方法入参提供自动注入的功能。@Resource要求提供一个Bean名称的属性,若属性为空,则自动采用标注处的变量名或方法作为Bean的名称。
@Component
public class Boss{
private Car car;
@Resource("car")
private void setCar(Car car){
this.car = car ;
}
}
@Autowired默认按类型匹配注入Bean,@Resource则按名称匹配注入Bean。而@Inject也是按类型匹配注入Bean的,只不过它没有required属性。
Spring为注解配置提供了一个@Scope的注解,通过它可以显示指定Bean的作用范围。
@Scope("prototype")
@Component
Spring支持JSR-250中定义的@PostConstruct和@PreDestroy注解,在Spring中他们相当于initmethod和destroy-method属性功能。但可以在一个Bean中定义多个@PostConstruce和@PreDestroy。
JavaConfig是Spring的一个子项目,它旨在通过Java类的方式提供Bean的定义信息。普通的POJO只要标注@Configuration注解,就可以为Spring容器提供Bean定义的信息了,每个标注了@Bean的类方法都相当于提供一个Bean的定义信息。
@Configuration
public class AppConf{
@Bean
public UserDao userDao(){
return new UserDao();
}
}
Spring提供一个AnnotationConfigApplicationContext类,它能够直接通过标注@Configuration的Java类启动Spring容器。
public class JavaConfigTest{
public static void main(String[] args){
ApplicationContext ctx = new AnnotationConfigApplicationContext(AppConf.class);
LogonService logonService = ctx.getBean(LogonService.class);
}
}
通过AnnotationConfigApplicationContext类的构造函数直接传入标注@Configuration的Java类,直接用该类中提供的Bean的定义信息启动Spring。此外,AnnotationConfigApplicationContext还支持通过编码的方式加载多个@Configuration配置类,然后通过刷新容器应用这些配置类。
public class JavaConfigTest{
public static void main(String[] args){
ApplicationContext ctx = new AnnotationConfigApplicationContext();
ctx.register(DaoConfi.class);
ctx.register(ServiceConfig.class);
ctx.refresh();
LogonService logonService = ctx.getBean(LogonService.class);
}
}
也可以通过@Import将多个配置类组装到一个配置类中。
@Configuration
@Import(DaoConfig.class)
public class ServiceConfig{
@Bean
public LogonService logonService(){
LogonService logonService = new LogonService();
return logonService;
}
}
在@Configuration配置类中可以通过@ImportResource引入XML配置文件,在配置类中即可直接通过@Autowired引用XML配置文件中定义Bean。
@Configuration
@ImportResource("classpath:xxx/xx/x.xml")
Bean不同配置方式的比较
Bean的定义
基于XML配置:在XML中通过<bean>元素定义Bean
基于注解配置:在Bean实现类出通过标注@Component或衍型类(@Service、@Repository、@Controller)定义Bean。
基于Java类配置:在标注了@Configuration的Java类中,通过在类方法上标注@Bean定义一个Bean。方法必须提供Bean的实例化逻辑。
Bean名称
基于XML配置:通过<bean>的id或name属性定义
基于注解配置:通过注解的value属性定义。默认名成为小写字母打头的类名
基于Java类配置:通过@Bean的name属性定义
Bean注入
基于XML配置:通过<property>子元素或通过p命名空间的动态属性
基于注解配置:通过在成员变量或方法入参出标注@Autowired,按类型匹配自动注入,还可配合使用@Qualifier按名称匹配注入
基于Java类配置:比较灵活,可以通过在方法处通过@Autowired使方法入参绑定Bean,然后在方法中通过代码进行注入,还可以通过调用配置类的@Bean方法进行注入
Bean生命过程方法
基于XML配置: 通过<bean>的init-method和destroy-method属性指定Bean实现类的方法名,最多只能指定一个初始化方法和一个销毁方法。
基于注解配置: 通过在目标方法上标注@PostContructor和@PreDestroy注解指定初始化和销毁方法,可以定义任意多个方法
基于Java类配置: 通过@Bean的initMethod或destroyMethod指定一个初始化或销毁方法。对于初始化方法来说,可以直接在方法内部通过代码的灵活定义初始化逻辑
Bean作用范围
基于XML配置: 通过<bean>的scope属性指定
基于注解配置: 通过在类定义处标注@Scope指定
基于Java类配置: 通过在Bean方法定义处标注@Scope指定
Bean延迟初始化
基于XML配置: 通过<bean>的lazy-init属性指定,默认为default,继承与<beans>的default-lazy-init设置,默认为false
基于注解配置: 通过在类定义处标注@Lazy指定
基于Java类配置: 通过在Bean方法定义处标注@Lazy指定
基于XML配置使用场合:Bean实现类来源于第三方类库,因无法在类中标注注解,通过XML配置方式较好;命名空间的配置,只能采用基于XML的配置
基于注解配置: Bean的实现类是当前项目开发的,可以直接在Java类中使用基于注解的配置
基于Java类配置: 基于Java类配置的又是在于可以通过代码方式控制Bean初始化的整体逻辑。若实例化Bean的逻辑比较复杂,则比较适合基于Java类配置的方式