Spring有哪些扩展点

一、ApplicationRunner和CommandLineRunner的拓展

实现接口ApplicationRunner和CommandLineRunner的类主要是在Spring容器启动之后做一些初始化的操作;

@Component@Order(value=1)

public class CommandLineApplication   implements   CommandLineRunner{

    @Override

    public void run(String...args)  throwsException{

    System.out.println(">>>>>>>>>>>>>>>>>>>开始启动后台任务>>>>>>>>>>>>>>>");

}


二、BeanFactoryPostProcessor和BeanPostProcessor的拓展

BeanFactoryPostProcessor和BeanPostProcessor都是针对容器内全部Bean实例进行自定义操作的,BeanFactoryPostProcessor可以获取BeanFactory,所以能够对bean的定义进行操作。BeanPostProcessor获取的是要加载到容器中的bean和beanName,其处理顺序在BeanFactoryPostProcessor之后。

@Component

@Order(value = 1)

public class CustomBeanFactoryPostProcessor implements BeanFactoryPostProcessor{

    @Override

    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {

    BeanDefinition beanDefinition = beanFactory.getBeanDefinition("myController");

    MutablePropertyValues values = beanDefinition.getPropertyValues();

    if (values.contains("name")) {

    values.addPropertyValue("age",18);

    }

}

BeanPostProcessor的常用场景是针对在加载Bean之前需要进行的通用操作。

@Component @Order(value = 1)

public class BeanPostProcessorTest implements BeanPostProcessor {

@Override

public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException

{

    System.out.println(">>>>>>>>>>>>>>BeanPostProcessorTest.postProcessBeforeInitialization()" + beanName);

    return bean;

}

}

三、Aware感知类拓展

aware是感知的意思,xxxAware就是能获取到xxx资源,实现xxxAware接口的setXXX()方法就能后获取xxx资源,再对其进行操作。

Aware接口在代码上实现BeanPostProcessor接口的postProcessBeforeInitialization方法。

常见Aware类和其用途:

BeanNameAware 获得到容器中Bean的名称

BeanFactoryAware 获得当前bean factory,这样可以调用容器的服务

ApplicationContextAware 获得当前application context,这样可以调用容器的服务

MessageSourceAware 获得message source这样可以获得文本信息

ApplicationEventPublisherAware 应用事件发布器,可以发布事件

ResourceLoaderAware 获得资源加载器,可以获得外部资源文件

BeanNameAware 获取到bean名称

EnvironmentAware 能够获取配置文件中的配置项

比如实现了EnvironmentAware接口的自定义类,在启动的时候会获取到application.properties中的配置项。

@Component @Order(value = 1)

public class EnvironmentAwareTest implements EnvironmentAware {

Environment environment;

@Override

public void setEnvironment(Environment environment) {

this.environment = environment;

System.out.println(">>>>>>>>>>>>>>>>>>>name = " + environment.getProperty("name"));

}

}

四、bean中方法级别拓展

主要说明bean在初始化前后的拓展点。主要有两种方法:

方法一:通过bean实现InitializingBean和 DisposableBean接口;

方法二:通过@PostConstruct 和 @PreDestroy 方法 实现初始化和销毁bean之前进行的操作;

针对方法一

通过实现InitializingBean接口的afterPropertiesSet()方法,看方法名即可得知是在bean属性赋值之后再进行该方法的处理;

针对方法二

只需要在需要在初始化时执行的方法上加注解@PostConstruct即可。

两种方法的实现代码如下:

@Component("initializingBeanTest")

@Order(value = 1)

public class InitializingBeanTest implements InitializingBean {

@Override

public void afterPropertiesSet() throws Exception {

//属性注入后进行处理

System.out.println(">>>>>>>>>>>InitializingBeanTest.afterPropertiesSet()");

}

@PostConstruct

void testPostConstruct() {

System.out.println(">>>>>>>>>>>>InitializingBeanTest.testPostConstruct()");

}

}

总结:这两者都是在bean赋值后初始化前进行的。

五、实现ApplicationContextInitializer接口的拓展

实现ApplicationContextInitializer这个类的主要作用就是在ConfigurableApplicationContext类型(或者子类型)的ApplicationContext做refresh之前,允许我们对ConfiurableApplicationContext的实例做进一步的设置和处理。常用场景是在开始创建所有bean之前做一些初始化操作。

@Component public class ApplicationContextInitializer1 implements ApplicationContextInitializer {

@Override

public void initialize(ConfigurableApplicationContext applicationContext) {

    // 打印容器里面有多少个bean

    System.out.println("bean count=====" + applicationContext.getBeanDefinitionCount());

     // 打印人所有beanName

    System.out.println(applicationContext.getBeanDefinitionCount() + "个Bean的名字如下:");

    String[] beanDefinitionNames = applicationContext.getBeanDefinitionNames();

    for (String beanName : beanDefinitionNames) {

    System.out.println(beanName); } } }

@SpringBootApplication

@EnableAsync

public class SpringDemoApplication {

     public static void main(String[] args) {

    SpringApplication app = new SpringApplication(SpringDemoApplication.class);

    // 方法一:添加自定义的 ApplicationContextInitializer 实现类的实例(注册ApplicationContextInitializer)                app.addInitializers(new ApplicationContextInitializer1());

    app.run(args);

    }

}

六、总结

以上各种拓展点的执行顺序大致如下:


参考资料

SpringBoot扩展点之二:ApplicationRunner和CommandLineRunner的扩展:https://www.cnblogs.com/duanxz/p/11

作者:yangnk

链接:https://www.zhihu.com/question/35226108/answer/3063756246

来源:知乎

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,701评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,649评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,037评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,994评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,018评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,796评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,481评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,370评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,868评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,014评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,153评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,832评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,494评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,039评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,156评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,437评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,131评论 2 356

推荐阅读更多精彩内容