Spring注解和配置方式

时间:2024-01-18 17:23:44

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类配置的方式