利用Spring的ApplicationEvent执行自定义方法
在Spring中已经定义了五个标准事件,分别介绍如下:
1)ContextRefreshedEvent:当ApplicationContext初始化或者刷新时触发该事件。
2)ContextClosedEvent:当ApplicationContext被关闭时触发该事件。容器被关闭时,其管理的所有单例Bean都被销毁
3) RequestHandleEvent:在Web应用中,当一个http请求(request)结束触发该事件
4)ContestStartedEvent:Spring2.5新增的事件,当容器调用ConfigurableApplicationContext的Start()方法开始/重新开始容器时触发该事件
5)ContestStopedEvent:Spring2.5新增的事件,当容器调用ConfigurableApplicationContext的Stop()方法停止容器时触发该事件
至于之后是否新增的暂时没研究
业务需求:项目启动时向数据库或者Redis初始化微信的access_token
新建SysInition(自定义名字)类,实现ApplicationListener接口,并监控ContextRefreshedEvent事件,在onApplicationEvent()方法中添加事件处理代码
import javax.annotation.Resource; import org.apache.commons.lang.StringUtils; import org.apache.log4j.Logger; import org.springframework.context.ApplicationListener; import org.springframework.context.event.ContextRefreshedEvent; import org.springframework.stereotype.Component; import com.phil.hi.service.auth.AuthTokenService; /** * 项目初始化操作类 * * @author phil * @date 2017年7月9日 * */ public class SysInition implements ApplicationListener<ContextRefreshedEvent> { private static final Logger logger = Logger.getLogger(SysInition.class); @Resource private AuthTokenService authTokenService; private static boolean flag = false; //防止二次调用 @Override public void onApplicationEvent(ContextRefreshedEvent event) { if (!flag) { flag= true; String token = authTokenService.findToken(); if (StringUtils.isBlank(token)) { logger.info("token is null in db"); authTokenService.saveToken(); } } } }
spring配置文件:
<bean id="sysInition" class="com.phil.hi.config.SysInition"></bean>
注解定义
在类前加上@Component或@Controller,然后在<context:component-scan base-package="">里加上你的类所在的包名
解决onApplicationEvent(方法被执行两次以上的问题
原因:无论是使用spring还是spring mvc,系统会存在两个容器,一个是root application context ,另一个就是projectName-servletContext(作为root application context的子容器)。这种情况下,就会造成onApplicationEvent方法被执行两次。
解决方法:如代码所示,用个标志标记是否已执行