解決spring定時任務執行兩次和tomcat部署緩慢的問題


spring定時任務執行兩次

問題重現和解析

最近使用quartz定時任務框架,結果發現開發環境執行無任何問題,部署到服務器上后,發現同一時間任務執行了多次。經過搜索發現是服務器上tomcat的配置文件出現了問題。
原來的配置文件——server.xml如下:

<Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="true">
	<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
		   prefix="localhost_access_log" suffix=".txt"
		   pattern="%h %l %u %t &quot;%r&quot; %s %b" />
</Host>
<Host name="www.xxx.com" appBase="webapps" unpackWARs="true" autoDeploy="true" xmlValidation="false" xmlNamespaceAware="false">
	<Context path="" docBase="/usr/local/tomcat/apache-tomcat-8.5.9/webapps/xxxindex"  reloadable="true"></Context>  
</Host>

一個Host表示一個容器,里面可以包含若干個Context(應用)。上面這段配置文件意思就是:在tomcat中配置了兩個容器,一個name=localhost,應用的根目錄為webapps,並且會自動解壓war包和自動部署。沒有指定context,會把根目錄下的所有web應用都部署,部署成功后,外網可以通過服務器IP+項目名來訪問;另一個name=www.xxx.com,和第一個host不同在於,配置了主頁web應用,且不需要跟項目名就可以訪問。部署成功后可以通過域名+項目名訪問,主頁所在項目可以直接通過根域名訪問。

這個時候問題就來了,包含定時任務的項目部署在webapps目錄下,tomcat中兩個獨立的容器都部署了一遍,相當於項目在服務器上的tomcat上部署了兩次,兩邊同時會運行定時任務,指定的是同一個數據庫。

問題解決

因此,為了盡可能不影響其他項目的正常訪問,我做了折中,講需要執行定時任務的項目單獨部署在另一個文件夾中,例如webroot ,然后只使用域名那個host,配置文件修改后如下:

<Host name="localhost" appBase="webapps" unpackWARs="true" autoDeploy="true">
	<Valve className="org.apache.catalina.valves.AccessLogValve" directory="logs"
		   prefix="localhost_access_log" suffix=".txt"
		   pattern="%h %l %u %t &quot;%r&quot; %s %b" />
</Host>

<Host name="www.xxx.com" appBase="" unpackWARs="true" autoDeploy="true" xmlValidation="false" xmlNamespaceAware="false">
	<Context path="" docBase="/usr/local/tomcat/apache-tomcat-8.5.9/webapps/xxxindex"  reloadable="true"></Context>  
	<Context path="/projectA" docBase="/usr/local/tomcat/apache-tomcat-8.5.9/webapps/projectA"  reloadable="true"></Context>  
	<Context path="/projectB" docBase="/usr/local/tomcat/apache-tomcat-8.5.9/webapps/projectB"  reloadable="true"></Context>  
	<Context path="/projectC" docBase="/usr/local/tomcat/apache-tomcat-8.5.9/webroot/projectC"  reloadable="true"></Context>  
</Host>

可以看到projectC是包含定時任務的項目。這樣部署成功后,除了該項目只能通過域名訪問之外,其余項目的訪問方式和之前保持不變。同時問題解決,定時任務只執行一次。

網上的另一種說法

<Host name="localhost"  appBase="webapps" unpackWARs="true" autoDeploy="true">  
	<Context docBase="projectA" path="" reloadable="true" />  
</Host>  

只有一個host,tomcat在啟動時,會部署一次根目錄下的所有項目,然后Context又會單獨部署一次,所以也會導致定時任務執行2次。
對於這種問題,解決的方案也有多種:

  1. 將huost的appBase設為空,將Context的Context 指向項目部署位置的絕對路徑。
  2. 刪除Context節點。

tomcat部署緩慢的問題

用的阿里雲服務器,部署tomcat時速度非常慢,但是后來買的新阿里雲又沒有這個問題。部署項目后一直會在

INFO [localhost-startStop-1] org.apache.catalina.startup.HostConfig.deployDirectory Deploying web application directory /opt/apache-tomcat-8.0.15-server/webapps/ROOT

這里卡好幾分鍾才會繼續下去。之前一直以為是服務器配置原因,后來無意中發現是jre的配置原因。參考了幾篇博客,發現oracle在WebLogic的文檔下Avoiding JVM Delays Caused by Random Number Generation給了原因和解決方案。

The library used for random number generation in Sun's JVM relies on /dev/random by default for UNIX platforms. This can potentially block the WebLogic SIP Server process because on some operating systems /dev/random waits for a certain amount of "noise" to be generated on the host machine before returning a result. Although /dev/random is more secure, BEA recommends using /dev/urandom if the default JVM configuration delays WebLogic SIP Server startup.

意思就是:

  1. JVM上產生隨機數的策略有兩種:/dev/random 和/dev/urandom。
  2. tomcat或者WebLogic等web服務器在部署時需要等待若一段隨機數產生的時間。unix平台下JVM默認采用的是安全性更好的/dev/random,但是潛在的會阻塞服務進程。
  3. 推薦使用/dev/urandom,產生隨機數速度快,/dev/random需要時間間隔生成隨機數,部署時間長。

修改方式:

  1. 打開$JAVA_HOME/jre/lib/security/java.security文件。
  2. securerandom.source=file:/dev/random 修改為securerandom.source=file:/dev/urandom

重啟tomcat,三十秒部署成功,solve it

參考文獻


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM