JavaWeb專案啟動時,自動執行程式碼的三種方式(包含不佔用tomcat啟動時長的方式)
阿新 • • 發佈:2018-11-12
三種方式實現在tomcat啟動時執行某段程式碼
由於這三種方式的執行時長計算在tomcat的啟動時長裡,如果tomcat設定了啟動超時時間,那麼這三種方式執行的操作很可能會讓tomcat啟動超時。
為了解決自動執行的部分不影響tomcat的正常啟動我們可以在三種方式中新建一個執行緒,將需要操作的部分交給子執行緒去做。
我們可以取三種方式的任意一種,新建一個執行緒:
public class GreyStartServlet extends HttpServlet {
@Override
public void init() throws ServletException {
MyThread read = new MyThread();
// 使用另一個執行緒來執行該方法,會避免佔用Tomcat的啟動時間
new Thread(thread).start();
}
}
class MyThread implements Runnable {
// Tomcat啟動結束後執行
@Override
public void run() {
// 子執行緒需要做的事情
}
}
1.ServletContextListener
web.xml配置
<listener>
<listener-class>com.yuan.framework.GreyClientInitListener</listener-class >
</listener>
1 public class GreyClientInitListener implements ServletContextListener {
2 private static final Logger LOGGER = LoggerFactory.getLogger(GreyClientInitListener.class);
3 public GreyClientInitListener() {}
4 public void contextDestroyed(ServletContextEvent arg0) {}
5 public void contextInitialized(ServletContextEvent arg0) {
6 try {
7 // 需要實現的功能
8 } catch (Exception e) {
9 LOGGER.error("GreyClientInitListener error", e);
10 }
11 }
12 }
2.HttpServlet
web.xml配置
<servlet>
<servlet-name>event-collector</servlet-name>
<servlet-class>com.yuan.framework.GreyStartServlet</servlet-class>
<load-on-startup>5</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>event-collector</servlet-name>
<url-pattern>/event-collect</url-pattern>
</servlet-mapping>
1 public class GreyStartServlet extends HttpServlet {
2 // Servlet的init方法會在Tomcat啟動的時候執行
3 @Override
4 public void init() throws ServletException {
5 // 需要實現的功能
6 }
7 }
3.spring ApplicationListener
1 @Service
2 public class StartGateServiceData implements ApplicationListener<ContextRefreshedEvent> {
3 private static final Log LOGGER = LogFactory.getLog(StartGateServiceData.class);
4 @Override
5 public void onApplicationEvent(ContextRefreshedEvent event) {
6 try {
7 // 在web專案中(spring mvc),系統會存在兩個容器,一個是root application context
8 // ,另一個就是我們自己的 projectName-servlet context(作為root application context的子容器)。
9 // 這種情況下,就會造成onApplicationEvent方法被執行兩次。為了避免這個問題,我們可以只在root
10 // application context初始化完成後呼叫邏輯程式碼,其他的容器的初始化完成,則不做任何處理。
11 if (event.getApplicationContext().getParent() == null) {
12 // 需要實現的功能
13 }
14 } catch (Exception e) {
15 LOGGER.error("StartGateServiceData", e);
16 }
17 }
18 }