zoukankan      html  css  js  c++  java
  • Spring Bean装配学习

    Spring Bean装配学习

    解释:所谓装配就是把一个类需要的组件给它设置进去,英文就是wire,wiring;注解Autowire也叫自动装配。

    目前Spring提供了三种配置方案:

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

    就我个人而言,用XML和自动装配混搭最多,用Java代码进行装配用的最少,几乎不用。这三种配置方案提供的功能会有重叠,大部分都可以根据个人喜好来选择。Spring的配置风格是可以相互搭配的,三种方式可以共存。

    三者的适用范围:

    XML > JavaConfig > 注解

    自动化装配bean

     自动化装配最为便利,写的东西最少,用起来很快。要实现自动化装配可以从两个方面来看:

    组件扫描(component scanning):Spring会自动发现应用上下文中所创建的bean

    自动装配(autowiring):Spring自动满足bean之间的依赖

     具体的步骤可以用下图来描述:

    创建可被发现的bean

    这里用CD播放器来先演示依赖注入(Dependency Inject)。

    涉及到的类图如下:

    CompactDisc接口

    1 package soundsystem;
    2 
    3 public interface CompactDisc {
    4     void play();
    5 }

    CompactDisk接口的实现类:SgtPeppers

    复制代码
     1 package soundsystem;
     2 
     3 import org.springframework.stereotype.Component;
     4 
     5 @Component
     6 public class SgtPeppers implements CompactDisc {
     7 
     8     private String title = "Sgt. Pepper's Lonely Hearts Club Band";
     9     private String artist = "The Beatles";
    10 
    11     public void play() {
    12         System.out.println("Playing " + title + " by " + artist);
    13     }
    14 }
    复制代码

    开启组件扫描的Java配置类:CDPlayerConfig

    复制代码
    1 package soundsystem;
    2 
    3 import org.springframework.context.annotation.ComponentScan;
    4 import org.springframework.context.annotation.Configuration;
    5 
    6 @Configuration
    7 @ComponentScan
    8 public class CDPlayerConfig {
    9 }
    复制代码

    测试类:CDPlayerConfig

    复制代码
     1 package soundsystem;
     2 
     3 import org.junit.Test;
     4 import org.junit.runner.RunWith;
     5 import org.springframework.beans.factory.annotation.Autowired;
     6 import org.springframework.test.context.ContextConfiguration;
     7 import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
     8 import org.springframework.util.Assert;
     9 
    10 @RunWith(SpringJUnit4ClassRunner.class)
    11 @ContextConfiguration(classes=CDPlayerConfig.class)
    12 public class CDPlayerTest {
    13 
    14     @Autowired
    15     private CompactDisc cd;
    16 
    17     @Test
    18     public void cdShouldNotBeNull() {
    19         Assert.notNull(cd, "inject failed");
    20     }
    21 }
    复制代码

    只需简单几步即可实现依赖注入,很强大。

    解释说明: 

    给SgtPeppers用了@Component这个注解后,Spring会为这个类去创建bean。

    组件扫描默认是不启用的,需要显式的配置Spring让其去寻找带有@Component注解的类,并为其创建bean。

    开启组件扫描的任务是CDPlayerConfig来实现的,通过Java代码定义了Spring的装配规则。

    如果没有其他的配置,@ComponentScan默认扫描与配置类相同的包。

    @CDPlayerConfig类位于soundsystem包中,Spring将会扫描这个包和这个包下的所有子包。

    如果用XML来开启组件扫描的话,可以使用<context:component-scan>元素:<context:component-scan base-package="soundsystem" />

    测试类中用到了两个注解:SpringJUnit4ClassRunner会在测试的时候自动创建Spring的应用上下文,@ContextConfiguration会告诉它需要在CDPlayerConfig类中加载配置,然后类CDPlayerConfig中包含了@ComponentScan,所以上下文中会包含CompactDisc的bean。

    为组件扫描的bean命名

    Spring应用上下文中所有的bean都会有一个ID。如果想之前的例子那样没有明确的给出bean的ID,Spring会根据类名为其指定一个ID。第一个类名小写。

    如果想为bean设置不同的ID,可以将期望的ID传给注解@Component。如下:

    复制代码
     1 package soundsystem;
     2 
     3 import org.springframework.stereotype.Component;
     4 
     5 @Component("lonelyHeartsClub")
     6 public class SgtPeppers implements CompactDisc {
     7 
     8     private String title = "Sgt. Pepper's Lonely Hearts Club Band";
     9     private String artist = "The Beatles";
    10 
    11     public void play() {
    12         System.out.println("Playing " + title + " by " + artist);
    13     }
    14 }
    复制代码

    另一种是使用Java依赖注入规范所提供的@Named注解来为bean设置ID(几乎没用过),大多数场景可以替换使用。

    可用的注解还有:@Service,@Repository等

    设置组件扫描的基础包

    对于包的扫描有以下几点可以记一下:

    不设置任何属性:配置类所在的包为基包,会以配置类所在的包作为基础包(base package)来扫描组件。

    只设置value:指定基包

    设置basePackages属性:更明确的指定了基包,而且给定的字符串作为基包

    设置basePackageClasses属性:明确地指定了类所在的包为基包

    空标记接口:可以将实际的应用代码和配置代码分开

    总结如下:

    复制代码
    // 啥属性没有, 就是以CDPlayerConfig所在的类为基包
    @Configuration
    @ComponentScan
    public class CDPlayerConfig { }
    

    // 指定了value属性, 以value指代的包为基包
    @Configuration
    @ComponentScan(
    "soundsystem")
    public class CDPlayerConfig { }

    // 明确指定了基包组
    @Configuration
    @ComponentScan(basePackages
    = "soundsystem")
    public class CDPlayerConfig { }

    // 指定了类所在的包为基包, 可以用一个标记接口替换实际的应用类
    @Configuration
    @ComponentScan(basePackageClasses
    = {CDPlayer.class, DVDPlayer.class)
    public class CDPlayerConfig { }

    复制代码

    为bean添加注解实现自动装配

    如果只是把类通过加上Component注解并进行了组件扫描来交给Spring管理,生成bean其实还不够。很多对象都会依赖其他对象协作完成任务。这样的话就需要一种方法能将组件扫描得到的bean和它们的依赖装配到一起,这就是自动装配。这里借助的是Spring的Autowired注解。

    @Autowire注解不仅能用在构造器上还可以用在属性的setter方法上(不仅仅是setter方法,Autowired可以用在类的任何方法上)。

    在Spring初始化了bean之后,它会尽可能地去满足bean的依赖。

    如果没有匹配的bean,那么在应用上下文创建的时候Spring会抛出一个异常,为了避免异常可以将Autowired的required属性设置为false。这样的话Spring会尝试执行自动装配,但是如果没有匹配的bean的话,Spring将会让这个bean处于未装配的状态(如果没有装配的话,使用的时候可能会报NullPointerException)。

    如果有多个bean都能满足依赖关系的话,Spring将会抛出一个异常,表明没有明确指定哪个bean来自动装配。

    @Autowired是Spring的独有注解,你还可以使用@Inject和@Resource。

    Java代码装配bean

    还可以用Java代码来做配置,之前只是用Hibernate的时候用过类进行配置,我用的也比较少。可能是因为喜欢吧配置归配置,代码归代码吧。

    组件扫描和自动装配的一个局限在于:有时候行不通。比如你要想在第三方的库中的组件装配到你的应用中,这种情况下是没法给它的类加上@Component和@Autowired注解的,这时候自动装配就用不了了。另一种就是用Java代码来配置。

    优点:强大、类型安全,对重构友好,和普通的Java代码一样

    注意:配置代码不应该侵入到业务逻辑代码中,最好是放在单独的包中,和其他的应用逻辑分开

    创建配置类

    创建JavaConfig类很简单:只需要为其添加@Configuration注解就可以了,就表明它是一个配置类。

    声明简单的bean

    要在JavaConfig中声明bean,需要编写一个方法,这个方法会创建所需类型的实例,然后给这个方法添加@Bean注解:

    1 @Bean
    2 public CompactDisc sgtPeppers() {
    3     return new SgtPeppers();
    4 }

    @Bean注解会告诉Spring这个方法会返回一个对象,该对象要注册为Spring应用上下文的bean。

    默认情况下,bean的ID和带有@Bean注解的方法名一样,可以为Bean加上name属性,或者修改方法名来设置ID。

    借助JavaConfig实现注入

    通过Java代码组装的方法也比较别致。

    1 @Bean
    2 public CDPlayer cdPlayer() {
    3     return new CDPlayer(sgtPeppers());
    4 }

    要注意这里的sgtPeppers()不是普通的方法,而是加了Bean注解的方法,Spring会拦截对这个方法的所有调用,并确保返回该方法所创建的bean,而不是每次进行实际的调用。(这个有点意思呀,拦截)。

    可以将SgtPeppers的实例注入到任意数量的其他bean之中。默认情况下Spring中的bean都是单例的。

    1 @Bean
    2 public CDPlayer cdPlayer(CompactDisc compactDisc) {
    3     return new CDPlayer(compactDisc);
    4 }

    上面是通常引用其他bean最佳的选择,因为它不会要求CompactDisc声明到同一个配置中,甚至不需要CompactDisc在JavaConfig中声明,只要Spring应用上下文中有就可以了。个人对于这种方法还是不太习惯。

    XML装配bean

    创建XML配置规范

    XML文件中要以<beans>为根元素

    声明一个简单的<bean>

    1 <bean class="soundsystem.SgtPeppers" />

    如果没有给定明确的ID,bean将会根据全限定的类名来进行命名。本例中将会是:"soundsystem.SgtPeppers#0",其中"#0"是一个计数形式,如果声明了一个另外的SgtPeppers,那么它自动的到的ID回事"soundsystem.SgtPeppers#1"。

    通产来讲最好的方法是借助id属性,为每一个bean设置一个合适的名字。

    借助构造器注入初始化的bean

    在XML中进行依赖注入的时候,往往有多种可选的配置方案和风格。

    1 <bean id="cdPlayer" class="soundsystem.CDPlayer">
    2     <constructor-arg ref="compactDisc" />
    3 </bean>

    如果不使用ref,而是使用value,则表示将字面量注入进去:

    1 <bean id="compactDisc" class="soundsystem.BlankDisc">
    2     <constructor-arg value="Sgt. Peppers's Lonely Hearts Club Band" />
    3     <constructor-arg value="The Beatles" />
    4 </bean>

    装配集合

    这个就不具体讲了,到了要用的时候来查一下就可以,list,set和数组都可以装配,就是set使用的时候重复的值会被忽略掉,而且不能保证顺序。

    设置属性

    1 <bean id="cdPlayer" class="soundsystem.CDPlayer">
    2     <property name="compactDisc" ref="compactDisc" />
    3 </bean>

    属性也可以注入字面量,集合。

    导入和混合配置

    以上的三种装配方案可以混合使用,而且自动装配的时候不会介意你的bean来自于哪里。

    JavaConfig导入其他的JavaConfig以及XML

    复制代码
    1 @Configuration
    2 @Import(CDConfig.class)
    3 public class CDPlayerConfig {
    4 
    5     @Bean
    6     public CDPlayer cdPlayer(CompactDisc compactDisc) {
    7         return new CDPlayer(compactDisc);
    8 
    9 }
    复制代码

    或者采用更高级别的配置类来导入:

    1 @Configuration
    2 @Import({CDPlayerConfig.class, CDConfig.class})
    3 public class SoundSystemConfig {
    4 
    5 }

    如果要导入XML配置的话:使用@ImportResource注解,使用相对于根类路径的地址

    1 @Configuration
    2 @Import({CDPlayerConfig.class})
    3 @ImportResource("classpath:cd-config.xml")
    4 public class SoundSystemConfig {
    5 
    6 }

    XML配置中引用JavaConfig

    XML可以用<import>标签导入其他的XML配置,用<bean class="soundsystem.CDConfig" />可以可以导入Java配置类。其实很简单的~

    总结

  • 相关阅读:
    tf.contrib.layers.fully_connected参数笔记
    关于RNN(Recurrent Neural Network)的一篇文章
    tf.contrib.rnn.LSTMCell 里面参数的意义
    机器学习笔记——k-近邻算法(一)简单代码
    labview2016崩溃解决方案
    tecplot 插值问题
    网络设备
    tcp
    udp
    icmp
  • 原文地址:https://www.cnblogs.com/jobs-lgy/p/9895022.html
Copyright © 2011-2022 走看看