Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 357|回复: 0

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

[复制链接]
  • TA的每日心情
    奋斗
    8 小时前
  • 签到天数: 773 天

    [LV.10]以坛为家III

    2045

    主题

    2103

    帖子

    71万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    713646
    发表于 2021-7-6 10:52:20 | 显示全部楼层 |阅读模式

    在做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.     }  
    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-8-9 16:59 , Processed in 0.064453 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表