Spring初始化完成后直接执行一个方法,初始化数据(解决方法被重复执行两次的情况)

在做WEB项目时,经常在项目第一次启动时利用WEB容器的监听、Servlet加载初始化等切入点为数据库准备数据,这些初始化数据 是系统开始运行前必须的数据,例如权限组、系统选项、默认管理员等等。但是项目若不是WEB工程,或者说还没用到WEB层(例如单元测试),这时应如何方 便地初始化数据呢?

    借助Spring容器是个很好的解决方案。Spring框架提供了事件机制,而事件机制必须实现ApplicationListener监听器,因此我们 只要编写一个实现类实现该接口的onApplicationEvent方法,在方法体中检测数据库的初始化数据是否存在并选择初始化之。

第一步:定义一个类InitData实现ApplicationListener

  1. import java.util.HashMap;  
  2. import java.util.Map;  
  3.   
  4. import org.apache.commons.logging.Log;  
  5. import org.apache.commons.logging.LogFactory;  
  6. import org.springframework.context.ApplicationEvent;  
  7. import org.springframework.context.ApplicationListener;  
  8. import org.springframework.stereotype.Controller;  
  9.   
  10. import com.ebay.cloud.cms.typsafe.metadata.model.MetaClass;  
  11. import com.ebay.tools.cms.exception.UtilityException;  
  12. import com.ebay.tools.cms.util.CommonInterfaceUtility;  
  13. import com.ebay.tools.cms.util.ExceptionUtils;  
  14. import com.ebay.tools.cms.util.LoggerUtil;  
  15.   
  16. /** 
  17.  * @author Josh Wang(Sheng) 
  18.  * @email swang6@email.com 
  19.  *  
  20.  * This class used to initialize data / load data ON the application started  
  21.  */  
  22. @Controller  
  23. public class InitData implements ApplicationListener<ApplicationEvent> {  
  24.       
  25.     private Log logger = LogFactory.getLog(InitData.class);  
  26.       
  27.     private static boolean isStart = false;  
  28.   
  29.     private static Map<String, MetaClass> cmsMetas = new HashMap<String, MetaClass>();  
  30.   
  31.     public void onApplicationEvent(ApplicationEvent event) {  
  32.         if (!isStart) {  
  33.             isStart = true;  
  34.             LoggerUtil.info(logger, "Start to load CMS Meta data");  
  35.               
  36.             try {  
  37.                 cmsMetas = CommonInterfaceUtility.getMetaClass();  
  38.             } catch (UtilityException e) {  
  39.                 LoggerUtil.error(logger, "Load Meta Class failed" + ExceptionUtils.getStackTraceMsg(e));  
  40.             }  
  41.               
  42.             LoggerUtil.info(logger,"End to load CMS Meta data");  
  43.               
  44.             LoggerUtil.info(logger,"Start to load Other data");  
  45.               
  46.             LoggerUtil.info(logger,"End to load Other data");  
  47.         }  
  48.           
  49.     }  

 第二步:解决onApplicationEvent(方法被执行两次以上的问题:

原因:

      在web 项目中(spring mvc),系统会存在两个容器,一个是root application context ,另一个就是我们自己的 projectName-servlet  context(作为root application context的子容器)。 这种情况下,就会造成onApplicationEvent方法被执行两次。

解决方法:

      如代码所示,只需要使用一个类变量isStart即可。

需要注意的是,一定要加Annotation @Controller,这样才表示是一个Spring的Bean(否则在相关Spring的xml中配置),才能被Spring容器处理。

第三步,写一个Spring JunitTest,注意相关的Annotation

需要注意的就是在类上面加入如下两个Annotation即可:

@ContextConfiguration(locations = "file:../validator-web/src/main/webapp/WEB-INF/cms-validator-servlet.xml")

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration用于加载spring的核心配置文件,因为这个操作是一个公共的操作,所以我们常常可以写在一个BaseTest中。

  1. @ContextConfiguration(locations = "file:../validator-web/src/main/webapp/WEB-INF/cms-validator-servlet.xml")  
  2. public class BaseTest {  
  3.   
  4.     /** 
  5.      * Set up the environment 
  6.      * @throws Exception 
  7.      */  
  8.     @BeforeClass  
  9.     public static void setUp() throws Exception {  
  10.                  
  11.     }  
  12.       
  13.     public static void print(String message) {  
  14.         System.out.println(message);  
  15.     }  
  16.   
  17.     public static void printHighlight(String message) {  
  18.         System.err.println(message);  
  19.     }  
  20.   
  21. }  

然后只需要在你的测试类上用这个Annotation:@RunWith(SpringJUnit4ClassRunner.class)

  1. @RunWith(SpringJUnit4ClassRunner.class)  
  2. public class TestCacheUtil extends BaseTest {  
  3.   
  4.     @Test  
  5.     public void getCMSMetaAttributeType() throws UtilityException {  
  6.   
  7.     }  
原文地址:https://www.cnblogs.com/shihaiming/p/5725466.html