Spring04

CAP7章节  Spring底层对BeanPostProcessor的使用

1,ApplicationContextAwareProcessor实现分析:

此类帮我们组建IOC容器,跟进ApplicationContextAwareProcessor我们发现, 这个后置处理器其实就是判断我们的bean有没有实现ApplicationContextAware 接口,并处理相应的逻辑,其实所有的后置处理器原理均如此.

那么怎么组建呢? 只需要实现 ApplicationContextAware 接口

步骤:

    1>, 新建Plane.java(Jeep.java复制一份即可)

         class Plane implements ApplicationContextAware

 

分析一下ApplicationContextAwareProcessor类的方法

a,在创建Plane对象,还没初始化之前, 先判断是不是实现了ApplicationContextAware接口,

如果是的话就调用invokeAwareInterfaces方法, 并给里面注入值;

b,进入invokeAwareInterfaces()方法,判断是哪个aware, 如果是ApplicationContextAware, 就将当前的bean转成ApplicationContextAware类型, 调用setApplicationContext(), IOC容器注入到Plane里去;

c,debug调用; 测试用例打断点测试

d,也可看debug调用栈来分析;

: debug可以打在ApplicationContextAwareProcessor处理器类的applyBeanPostProcessorsBeforeInitialization()方法里, 方便调试, beanPlane类型时,F5跟进看, 最终在 InvokeAwareInterfaces()方法里返回我们的IOC容器applicationContext.

 

2,BeanValidationPostProcess分析:数据校验,

   BeanPostProcessor接口实现CTRL+T

了解即可,处理器的原理和其它处理器一致.

当对象创建完,给bean赋值后,在WEB用得特别多;把页面提交的值进行校验

3,InitDestroyAnnotationBeanPostProcessor

此处理器用来处理@PostConstruct, @PreDestroy, 怎么知道这两注解是前后开始调用的呢, 就是 InitDestroyAnnotationBeanPostProcessor这个处理的

@PostConstruct为例, 为什么声明这个注解后就能找到初始化init方法呢?

 

  

总结: Spring底层对BeanPostProcessor的使用, 包括bean的赋值, 注入其它组件, 生命周期注解功能,@Async, 等等

CAP8章节  @Value赋值

新建cap8目录;

1, 新建Bird.java

 

2,新建Cap8MainConfig.java配置类

3,新建测试用例Cap8Test.java, 从容器获取bean并打印

打印结果如下: 主要是没设值

4,以前使用bean.xml配置文件进行赋值

5,使用@Value赋值如何赋值呢?见下

6,从配置文件[properties]读取, 新建test.properties

7,test.properties内容为bird.color=red

8,Bird类新增private String colorsetget方法;

9,test.properties配置文件加载起来

10,再运行test01()用例, 打印出以下结果

11,test.properties值是加在运行环境变量里:

CAP9章节  @Autowired自动装配

自动装配:spring利用依赖注入(DI), 完成对IOC容器中的各个组件的依赖关系赋值

1,新建TestController.java   TestService.java  TestDao; 分别建在指定的包内,可看步骤2.

这些所有JAVA 类的对象扫描后都是保存在IOC容器中管理的;

          

2,新建配置类Cap9MainConfig.java(),扫描并将以上bean都扫描并加载到容器

3, 针对以上基础类建立完成后, 可以先做个测试

TestService.java, 使用Autowired注入,并把testDao打印出来(在测试时方便对比)

4, 新建Cap9Test.java测试用例,比较TestService拿到testDao与直接从容器中拿到的testDao是否为同一个?

结果很明显是同一个testDao,地址一样

小结:

@Autowired表示默认优先按类型去容器中找对应的组件,相当于anno.getBean(TestDao.class)去容器获取idtestDaobean, 并注入到TestServicebean;

使用方式如下:

   TestService{

        @Autowired

        private TestDao testDao;//默认去容器中找id”testDao”bean

   }

5, 注意事项

5.1如果容器中找到多个testDao, 会加载哪个testDao?

   操作步骤:

Cap9MainConfig.java声明@Bean(“testDao2”)

并将TestDao加入flag属性和set, gettoString方法,用来分辨加载了哪个bean.

如何区分TestService是使用了(@ReponstrytestDaoflag=1)bean还是(testDao2flag=2)bean?

测试步骤如下:

1,直接使用@Autowired, testDao注入到TestService

测试结果

 

2,如果一定要使用容器中的testDao2?操作如下:

测试结果

3,虽然以上定义了private TestDao testDao2, 但还是想加载bean idtestDao(flag=1)bean,怎么办?此时可以使用@Autowired@Qualifier结合来指定注入哪一个bean,

操作如下:

测试结果

4,如果容器中没有任何一个testDao, 会出现什么状况呢?

操作如下: 注释掉@Repository@Bean("testDao2")

此时容器启动时这两个bean都不会加载(因为注解被注释啦.......)

测试结果如下:

很明显报错了, 因为@Autowired注解里的属性默认required=true.必须找到bean

那怎么解决呢?

测试结果如下:

 

5,@Primary注解指定bean如何加载呢?

(:将以上原注释掉的@Repository@Bean("testDao2") 恢复,见下图)

重要:为了验证@Qualifier@Primary两注解的加载顺序,测试如下

当对于testDao在容器中同时存在多个时, @Qualifier@Primary注解同时存在,会发生什么呢?

见下操作:  打开@Qualifier@Primary注解.

测试结果:

此时只能说明一点: @Qualifier是根据bean id指定获取testDao, 不受@Primary影响.

那么@Primary的功能在哪呢?继续测试.....

 

测试结果:

5.2除了@Autowired, 是不是还用过@Resource(JSR250)  @Inject(JSR330)

QualifierAutowired注释掉(注意: 此时@Primary 还没注释......)

 

测试结果:

效果也是一样的, 但它不先优先装配@Primarybean

5.3 @Inject自动装配的使用:

:@Inject@Autowired的区别如下:

@InjectAutowired一样可以装配bean, 并支持@Primary功能, 可用于非spring框架.

@Inject缺点: 但不能支持@Autowired(required = false)的功能,需要引入第三方包javax.inject

操作步骤:

1,pom.xml导入javax.inject

2,使用@Inject注解

结论:@Inject不支持required=false,  但支持primary

 

Autowired属于spring, 不能脱离spring,  @Resource@Inject都是JAVA规范

推荐大家使用@Autowired

原文地址:https://www.cnblogs.com/keiyoumi520/p/14702703.html