zoukankan      html  css  js  c++  java
  • Spring在Web容器启动时执行初始化方法

    需求:在tomcat启动时开启一个定时任务。

    想法:容器启动时执行方法,最容易想到的就是servlet中可以配置load-on-startup,设置一个正整数也就可以随容器一起启动。

    问题:上面的方法很好,但是由于定时任务需要去操作数据库,而项目采用了spring的依赖注入来管理对象,而servlet并不受Spring的管理。若此时在servlet中注入Spring管理的对象,则会报错:javax.naming.NameNotFoundException: Name com.test.InitServlet is not bound in this Context。

    所以想要再servlet中操作数据库,只能手动去创建一个service,这显然违背了我们使用Spring的初衷,让项目看起来不伦不类的。那么如何才能在启动WEB容器的时候执行一段初始化代码,并且可以让其被Spring管理呢?

    解决方案:Spring提供了当一个Bean初始化后执行方法的扩展点:InitializingBean。这里的初始化指的就是当该Bean的属性被注入完成后(注意:这里并不是所有属性都需要被设置),所以InitializingBean接口提供的方法名称也很形象:afterPropertiesSet()。

    使用的时,将该Bean注入到Spring容器,之后我们便可以获得Spring容器中的对象了,也就是说,可以得到service方法来执行我们的定时任务了。

    具体代码如下:

    [java] view plain copy
    1. @Component  
    2. public class InitServlet implements InitializingBean {  
    3.   
    4.     /** 
    5.      *  
    6.      */  
    7.     private static final long serialVersionUID = 1L;  
    8.       
    9.     @Resource  
    10.     private DispatchesService dispatchesService;  
    11.   
    12.     @Override  
    13.     public void afterPropertiesSet() throws Exception {  
    14. dispatchesService.spyDDetails();  
    15.     }  
    16.   
    17. }  

    另外还有两种方法也可以实现如上的功能。

    1、若采用XML来配置Bean的话,可以指定属性init-method

    2、通过注解@PostConstruct来修改初始化方法

    值得注意的是,三者可以同时存在,触发的顺序是先触发@PostConstruct修饰的方法,再触发afterPropertiesSet(),最后触发init-method

    其中@PostConstruct是通过注册一个BeanPostProcessor,在Bean的初始化方法之前调用,而afterPropertiesSet()和init-method都在初始化方法中调用

    关于@PostConstruct详细的介绍可以看这里:http://blog.csdn.net/yaerfeng/article/details/8447530

    下面是Spring中调用Bean的初始化代码的源代码:

    [java] view plain copy
    1.     protected void invokeInitMethods(String beanName, final Object bean, RootBeanDefinition mbd)  
    2.             throws Throwable {  
    3.   
    4.         boolean isInitializingBean = (bean instanceof InitializingBean);  
    5.         if (isInitializingBean && (mbd == null || !mbd.isExternallyManagedInitMethod("afterPropertiesSet"))) {  
    6.             if (logger.isDebugEnabled()) {  
    7.                 logger.debug("Invoking afterPropertiesSet() on bean with name '" + beanName + "'");  
    8.             }  
    9.             if (System.getSecurityManager() != null) {  
    10.                 try {  
    11.                     AccessController.doPrivileged(new PrivilegedExceptionAction<Object>() {  
    12.                         public Object run() throws Exception {  
    13.                             ((InitializingBean) bean).afterPropertiesSet();// 这里触发afterPropertiesSet
    14.                             return null;  
    15.                         }  
    16.                     }, getAccessControlContext());  
    17.                 }  
    18.                 catch (PrivilegedActionException pae) {  
    19.                     throw pae.getException();  
    20.                 }  
    21.             }                 
    22.             else {  
    23.                 ((InitializingBean) bean).afterPropertiesSet();// 这里触发afterPropertiesSet  
    24.             }  
    25.         }  
    26.   
    27.         if (mbd != null) {  
    28.             String initMethodName = mbd.getInitMethodName();// 这里是触发init-method  
    29.             if (initMethodName != null && !(isInitializingBean && "afterPropertiesSet".equals(initMethodName)) &&  
    30.                     !mbd.isExternallyManagedInitMethod(initMethodName)) {  
    31.                 invokeCustomInitMethod(beanName, bean, mbd);  
    32.             }  
    33.         }  
    34.     }  


    补充:

    还有一种方法,是当Spring将所有的Bean都初始化完成后,会留给我们一个入口,我们可以实现如下接口

    [java] view plain copy
    1. @Component  
    2. public class InstantiationTracingBeanPostProcessor implements  
    3.         ApplicationListener<ContextRefreshedEvent> {  
    4.   
    5.     @Override  
    6.     public void onApplicationEvent(ContextRefreshedEvent arg0) {  
    7.         System.out.println("-----所有Bean载入完成---");  
    8.     }  
    9. }  
  • 相关阅读:
    Java 中的JOption函数
    01背包与完全背包(对比)
    AC注定不平坦(大神回忆录)
    背包精讲之——01背包
    动规问题概述(待整理)
    背包九讲
    Tautology(递推)||(栈(stack))(待整理)
    深度优先和广度优先区别
    Linux下JDK、Tomcat的安装及配置
    同IP不同端口Session冲突问题
  • 原文地址:https://www.cnblogs.com/molashaonian/p/8569921.html
Copyright © 2011-2022 走看看