Spring提供了很多切面,用于在项目启动的不同阶段植入代码。

BeanPostProcessor :可以在Bean创建之后,在初始化之前、初始化之后,进行一些额外的操作。

InitializingBean:在所有的Bean互相注入和Properties参数设置之后,在初始化函数调用之前,进行一些额外的操作。

BeanFactoryPostProcessor :在所有的Bean初始化之前,进行一些额外的操作,例如:手动注册对象到Spring容器。

具有相似功能的切面还有:

ContextLoaderListener:因为本身就是侦听项目启动和关闭的,直接在启动函数中添加后置代码也可以。

Bean的后处理

public class Person implements InitializingBean{
private String name; public Person() {
System.out.println("创建Person实例");
} public void setName(String name) {
System.out.println("Set方式设值:"+name);
this.name = name;
} public void init(){
System.out.println("Person初始化:init执行");
} @Override
public String toString() {
return "Person [name=" + name + "]";
} @Override
public void afterPropertiesSet() throws Exception {
System.out.println("在Bean后处理之后:afterPropertiesSet执行");
}
}
/**
* Bean后处理器
* @author ChenSS
* @2016年12月30日
*/
public class MyBeanPostProcessor implements BeanPostProcessor { @Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
System.out.println("Bean后处理postProcessBeforeInitialization之后:init之前");
return bean;
} @Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
System.out.println("Bean后处理postProcessAfterInitialization:init之后");
if (!(bean instanceof Person))
return bean;
Person person = (Person) bean;
person.setName("新名字:xiaohua");
return person;
}
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
<bean id="xiaoming" class="com.spring.test.Person" init-method="init">
<property name="name" value="小明" />
</bean>
<!-- 后处理器 -->
<bean id="myBeanPostProcessor" class="com.spring.test.MyBeanPostProcessor" />
</beans>
public class Test {
public static void main(String[] args) {
new ClassPathXmlApplicationContext("applicationContext.xml");
}
}

容器后处理器

容器后处理器,是一个大的切面,在每一个Bean创建之后,但是在初始化之前;

注意:Spring中XML解析是顺序执行的,如果提前注册了容器后处理器,后处理器对后续Bean无效(不严谨的说法,未做全面测试,但是确实反生过此Bug)。

定义一个最简单的容器后处理器

/**
* 容器后处理器,实现BeanFactoryPostProcessor接口
* @author ChenSS
* @2016年12月30日
*/
public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
System.out.println("容器后处理器:postProcessBeanFactory");
System.out.println(beanFactory.getClass().getName());
}
}

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
<bean id="xiaoming" class="com.spring.test.ioc.Person" init-method="init">
<property name="name" value="小明" />
</bean>
<!-- 后处理器 -->
<bean class="com.spring.test.ioc.MyBeanFactoryPostProcessor" />
</beans>

代码本身没什么意义,只是打印一下日志,让大家看一下效果,可见容器的后处理器,是在所有Bean创建之后,但是在初始化之前执行的。

附:PropertyPlaceholderConfigurer读取dp.properties配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<list>
<value>db.properties</value>
<!-- <value>other value...</value> -->
</list>
</property>
</bean>
<bean id="dataSource" class="com.spring.test.ioc.DateBaseProperties">
<!-- 以下这些值均来自配置文件dp.properties -->
<property name="driverClass" value="${jdbc.driverClass}" />
<property name="jdbcUrl" value="${jdbc.url}" />
<property name="user" value="${jdbc.user}" />
<property name="password" value="${jdbc.password}" />
</bean>
</beans>

准备dp.properties文件,这个文件也是放在Src根目录下

jdbc.driverClass=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/testdb
jdbc.user=root
jdbc.password=123456

对应的测试用JavaBean。

public class DateBaseProperties {
private String driverClass;
private String jdbcUrl;
private String user;
private String password; //set、toString方法省略
}

PropertyOverrideConfigurer读取dp.properties配置文件

看着前面的配置文件,总觉得怪怪的,不是在dp.properties配置了一次嘛?为什么还要在XML再配置一次?

Spring框架自带的加载工具类PropertyOverrideConfigurer可以帮我们解决这个问题。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
<bean class="org.springframework.beans.factory.config.PropertyOverrideConfigurer">
<property name="locations">
<list>
<value>db.properties</value>
<!-- <value>other value...</value> -->
</list>
</property>
</bean>
<!-- 这里的代码全删了 -->
<bean id="dataSource" class="com.spring.test.ioc.DateBaseProperties"/>
</beans>

不过关于db.properties的配置得调整一下,把前面的jdbc改成dataSource,名字和Bean的id对应。

dataSource.driverClass=com.mysql.jdbc.Driver
dataSource.url=jdbc:mysql://localhost:3306/testdb
dataSource.user=root
dataSource.password=123456

上面两个容器后处理器的测试结果如下

下面例举了一些我看到的、查到的其他后处理器:

CustomAutowireConfigurer自定义自动装配的配置器
CustomScopeConfigurer自定义作用于的配置器
CustomEditorConfigurer:此类注册一个PropertyEditor实现,该实现用户将配置文件中的字符串值转换为bean需要的类型。
ServletContextPropertyPlaceholderConfigurer:此回调处理器泛化了PropertyPlaceholderConfigurer类。因此,只要bean属性遵照指定命名规范,他就替换bean的属性。除了他的超类,此处理器还将从处理应用程序的servlet上下文参数入口装载值。
PreferencesPlaceholderConfigurer:此回调处理器将JDK1.4Preperences API替换bean属性中的值,此Preperences API标识他将解析来自用户Preperences的值,然后系统Preperences获得值,最后从一个引用文件获得值。

afterPropertiesSet

Spring框架——后处理器的更多相关文章

  1. Spring的后处理器-BeanPostProcessor跟BeanFactoryPostProcessors

    最近在重读spring源码(为什么要重读?因为不得不承认,去年跟着<深入解析sping源码>一书过了一遍spring的源码,除了满脑袋都是各种BeanFactory跟BeanDefinit ...

  2. Spring Bean后处理器以及容器后处理器【转】

    Bean后处理器:即当spring容器实例化Bean实例之后进行的增强处理. 容器后处理器:对容器本身进行处理,并总是在容器实例化其他任何Bean之前读取配置文件的元数据并可能修改这些数据. 一.Be ...

  3. Spring.net 后处理器 可用来切换实例

    .xml配置 <!--我们在Object.xml文件上将HexuObjectPostProcessor注册到上下文对象中去--> <object id="hexu" ...

  4. Spring框架简单介绍

    原文地址:  http://my.oschina.net/myriads/blog/37922 1.使用框架的意义与Spring的主要内容 随着软件结构的日益庞大,软件模块化趋势出现,软件开发也须要多 ...

  5. JavaWeb_(Spring框架)在Struts+Hibernate框架中引入Spring框架

    spring的功能:简单来说就是帮我们new对象,什么时候new对象好,什么时候销毁对象. 在MySQL中添加spring数据库,添加user表,并添加一条用户数据 使用struts + hibern ...

  6. 吴裕雄--天生自然JAVA SPRING框架开发学习笔记:Spring DI(依赖注入)的实现方式属性注入和构造注入

    依赖注入(Dependency Injection,DI)和控制反转含义相同,它们是从两个角度描述的同一个概念. 当某个 Java 实例需要另一个 Java 实例时,传统的方法是由调用者创建被调用者的 ...

  7. 8 -- 深入使用Spring -- 1...两种后处理器

    8.1 两种后处理器 Spring框架提供了很好的扩展性,出了可以与各种第三方框架良好整合外,其IoC容器也允许开发者进行扩展,这种扩展甚至无须实现BeanFactor或ApplicationCont ...

  8. XI.spring的点点滴滴--IObjectFactoryPostProcessor(工厂后处理器)

    承接上文 IObjectFactoryPostProcessor(工厂后处理器)) 前提是实现接口的对象注册给当前容器 直接继承的对象调用这个postProcessBeanFactory方法,参数为工 ...

  9. Ⅹ.spring的点点滴滴--IObjectPostProcessor(对象后处理器)

    承接上文 IObjectPostProcessor(对象后处理器) 前提是实现接口的对象注册给当前容器 C#: 通过继承AbstractObjectFactory对象的AddObjectPostPro ...

随机推荐

  1. WPF DataGridHyperlinkColumn

    为了点击链接,我们使用 <DataGrid x:Name="data" LoadingRow="load" ItemsSource="{Bind ...

  2. Linq标准查询操作符

     Linq的出现让代码简洁了不少.之前在项目中基本都在使用它,但是没有完整的整理过,今天借这个周末,将其进行整理,方便后期对其的使用.Linq的操作可以分为聚合,连接,转换,元素操作符,相等操作,生成 ...

  3. LINUX 笔记-scp命令

    从本地服务器复制到远程服务器: (1) 复制文件: 命令格式: scp local_file remote_username@remote_ip:remote_folder (2) 复制目录: 命令格 ...

  4. JAVA基础知识总结:六

    一.不定长参数 1.语法:数据类型... 变量名称 使用注意事项:a.不定长参数就相当于是一个数组 b.不定长参数只能出现在参数列表的最后面 c.一个函数的参数列表中只能出现一次不定长参数 d.对于不 ...

  5. highcharts框架使用总结

    Highcharts官网地址:https://www.hcharts.cn/products/highcharts首先需要引入jQuery框架,然后包含Highcharts框架需要使用到的js文件,最 ...

  6. 2_ROS学习

    2_VNC远程连接树莓派 在上一次,我们成功的给树莓派安装了Ubuntu mate的操作系统. 树莓派是嵌入式计算机,一般是没有显示屏来显示的,我们通过远程连接来访问树莓派.网上推荐了ssh连接,xr ...

  7. 微信公众号开发(十二)OAuth2.0网页授权

    OAuth允许用户提供一个令牌,而不是用户名和密码来访问它们存放在特定服务器上的数据,每一个令牌授权一个特定的网站在特定时段内访问特定的资源. 授权过程如下: 1.引导用户进入授权页面同意授权,获取c ...

  8. 京东口红top 30分析

    一.抓取商品id 分析网页源码,发现所有id都是在class="gl-item"的标签里,可以利用bs4的select方法查找标签,获取id: 获取id后,分析商品页面可知道每个商 ...

  9. Lua 5.3 协程简单示例

    Lua 5.3 协程简单示例 来源 http://blog.csdn.net/vermilliontear/article/details/50547852 生产者->过滤器->消费者 模 ...

  10. Windows环境下多线程编程原理与应用读书笔记(4)————线程间通信概述

    <一>线程间通信方法 全局变量方式:进程中的线程共享全局变量,可以通过全局变量进行线程间通信. 参数传递法:主线程创建子线程并让子线程为其服务,因此主线程和其他线程可以通过参数传递进行通信 ...