java啟動(dòng)參數(shù)之謎的排查過程
背景
最近遇到一個(gè)有意思的事情,java應(yīng)用運(yùn)行在阿里云的ack集群中,某一天有個(gè)應(yīng)用啟動(dòng)突然發(fā)現(xiàn)阿里云上的agent都沒有注冊(cè)了,于是開始排查原因。
排查過程
我們的應(yīng)用是java應(yīng)用,jdk版本是Open-jdk8,阿里云agent是直接注入到容器中的,因此會(huì)將agent啟動(dòng)參數(shù)自動(dòng)注入到 JAVA_TOOL_OPTIONS 環(huán)境變量中,當(dāng)應(yīng)用啟動(dòng)時(shí)會(huì)自動(dòng)帶上agent啟動(dòng)參數(shù)。
agent沒注冊(cè),首先檢查應(yīng)用的啟動(dòng)日志,發(fā)現(xiàn)應(yīng)用是啟動(dòng)成功的,tomcat端口都是正常的。仔細(xì)觀察日志,發(fā)現(xiàn)了問題。由于agent 啟動(dòng)參數(shù)是注入到 JAVA_TOOL_OPTIONS 中的,通常jvm 在啟動(dòng)的時(shí)候會(huì)優(yōu)先加載 JAVA_TOOL_OPTIONS,日志中會(huì)出現(xiàn) Picked up JAVA_TOOL_OPTIONS
的字樣,如下圖所示,但是問題現(xiàn)場(chǎng)卻沒有這一行和agent相關(guān)的啟動(dòng)日志,說明 jvm 啟動(dòng)的時(shí)候并沒有加載 JAVA_TOOL_OPTIONS。
我們開始懷疑是 agent啟動(dòng)參數(shù) 的問題,以為是agent在容器重建時(shí)沒有將啟動(dòng)參數(shù)注入到環(huán)境變量中。但是通過環(huán)境變量一看,發(fā)現(xiàn) JAVA_TOOL_OPTIONS 是在的,而且每個(gè)agent的參數(shù)都是齊全的。
這個(gè)時(shí)候就開始懷疑是不是啟動(dòng)腳本的問題,是不是有人在啟動(dòng)腳本中加了unset JAVA_TOOL_OPTIONS
,因?yàn)楫?dāng)存在JAVA_TOOL_OPTIONS時(shí),使用jdk相關(guān)的命令都會(huì)帶上JAVA_TOOL_OPTIONS中的參數(shù),造成一定的困擾,所以有時(shí)候在排查問題的時(shí)候會(huì)先unset掉這個(gè)變量,但是檢查完腳本也沒有問題。
最后開始咨詢阿里云的工程師,懷疑是不是agent或者容器環(huán)境有問題。經(jīng)過反復(fù)比較正常容器和問題容器的JAVA_TOOL_OPTIONS啟動(dòng)參數(shù),發(fā)現(xiàn)問題容器因?yàn)槎嗉虞d一個(gè)agent,JAVA_TOOL_OPTIONS多出來一段參數(shù),去掉這段參數(shù)就能恢復(fù)正常,加上就會(huì)有問題。到這里,可能正常的思路都是懷疑是多出來的參數(shù)造成的。但在排查其他正常容器時(shí)發(fā)現(xiàn),有的容器即使有這一段參數(shù)也能正常啟動(dòng)。
這個(gè)時(shí)候,阿里云的工程師懷疑是不是參數(shù)太長(zhǎng)導(dǎo)致的,因?yàn)橛袉栴}的容器的應(yīng)用名字比較長(zhǎng),于是我們開始測(cè)試,發(fā)現(xiàn)確實(shí)是這個(gè)問題,如下圖所示。隨后確定了問題所在,jdk8 在加載默認(rèn)環(huán)境變量時(shí)會(huì)檢查長(zhǎng)度,當(dāng)大于1024字節(jié)時(shí)就會(huì)加載失敗。
環(huán)境變量
在jdk相關(guān)的環(huán)境變量中,有兩種默認(rèn)的環(huán)境變量 JAVA_TOOL_OPTIONS
和 _JAVA_OPTIONS
。
JAVA_TOOL_OPTIONS:在jdk8及之前版本中,該變量是最標(biāo)準(zhǔn)的,所有虛擬機(jī)都能識(shí)別和應(yīng)用的環(huán)境變量,在jdk9之后被JDK_JAVA_OPTIONS
所取代。該變量限制1024字節(jié),在不同虛擬機(jī)中表現(xiàn)不一樣,有的是加載失敗,有的是截取一段。
_JAVA_OPTIONS:也是默認(rèn)的環(huán)境變量,但是它是JVM廠家自定義的,可以覆蓋JAVA_TOOL_OPTIONS,但各廠家的命名不同,_JAVA_OPTIONS是Oracle的JVM,而IBM的則是用IBM_JAVA_OPTIONS。
因此為避免出現(xiàn)問題,我們應(yīng)該盡量避免使用默認(rèn)的環(huán)境變量,通常情況下可以在腳本中自定義啟動(dòng)變量如 JAVA_OPTS
、SPRINGBOOT_OPTS
等等。然后在啟動(dòng)java時(shí)顯式的指定啟動(dòng)參數(shù)。
java [-options] -jar xxx.jar [args…] 可以寫成 JAVA_OPTS="[-options]" JAVA_ARGS="[args…]" java ${JAVA_OPTS} -jar xxx.jar ${JAVA_ARGS}
附:?jiǎn)⑼D_本
項(xiàng)目打包后在測(cè)試環(huán)境的啟停都是個(gè)體力活,剛好又給筆者遇到了,綜合別人的腳本記錄了一下
判斷 Java 進(jìn)程是否存在
APP_NAME=xxx.jar pid=jps -l | grep $APP_NAME if [ -z $pid ]; then echo "$APP_NAME started" else echo "$APP_NAME stoped" fi # 普通進(jìn)程的 # pid=ps -ef | grep $APP_NAME | grep -v grep | awk '{print $2}'
啟停腳本
APP_NAME=xxxx-1.0-SNAPSHOT.jar pid=0 checkpid() { javaps=`jps -l | grep $APP_NAME` if [ -n "$javaps" ]; then pid=`echo $javaps | awk '{print $1}'` else pid=0 } start() { checkpid if [ $psid -ne 0 ]; then echo "$APP_NAME already started" else echo "Starting $APP_NAME ..." `nohup java -jar $APP_NAME > $APP_NAME'.out' 2>&1 &` checkpid if [ $pid -ne 0]; then echo "$APP_NAME start success" else echo "$APP_NAME start faild" fi fi } stop() { checkpid if [ $pid -ne 0 ]; then echo "Stoping $APP_NAME..." kill -9 $pid if [$? -eq 0 ]; then echo "$APP_NAME stop success" else echo "$APP_NAME stop faild" fi else echo "$APP_NAME already stoped" fi } case "$1" in 'start') start ;; 'stop') stop ;; 'restart') stop start ;; *) # 其他任何情況 echo "help: $0 {start|stop|restart}" echo "例子: ./deploy start exit 1 esac exit 0
總結(jié)
到此這篇關(guān)于java啟動(dòng)參數(shù)之謎的文章就介紹到這了,更多相關(guān)java啟動(dòng)參數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Eclipse中Properties和yml配置文件注釋亂碼的解決
這篇文章主要介紹了Eclipse中Properties和yml配置文件注釋亂碼的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10java中加密的實(shí)現(xiàn)方法(MD5,MD2,SHA)
這篇文章主要介紹了java中加密的實(shí)現(xiàn)方法(MD5,MD2,SHA)的相關(guān)資料,這里提供三種實(shí)現(xiàn)加密的方法,大家可以對(duì)比一下,需要的朋友可以參考下2017-08-08Java實(shí)現(xiàn)圖章或簽名插在pdf的固定位置
使用Java技術(shù)在word轉(zhuǎn)換成pdf過程中實(shí)現(xiàn)將圖章或者簽名插入在pdf中,并生成帶圖章或者簽名的pdf,來完成某些特定場(chǎng)景的需求,文中有詳細(xì)的代碼示例,需要的朋友可以參考下2023-10-10JDBC實(shí)現(xiàn)學(xué)生管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了JDBC實(shí)現(xiàn)學(xué)生管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-02-02使用Java通過OAuth協(xié)議驗(yàn)證發(fā)送微博的教程
這篇文章主要介紹了使用Java通過OAuth協(xié)議驗(yàn)證發(fā)送微博的教程,使用到了新浪微博為Java開放的API weibo4j,需要的朋友可以參考下2016-02-02SpringBoot單元測(cè)試之?dāng)?shù)據(jù)隔離詳解
我們?cè)趯憜卧獪y(cè)試時(shí),有一個(gè)比較重要的要求是可以重復(fù)運(yùn)行, 那么這樣就會(huì)有一個(gè)比較麻煩的問題:數(shù)據(jù)污染,所以本文為大家整理了兩個(gè)數(shù)據(jù)隔離的方式,希望對(duì)大家有所幫助2023-08-08spring+mybatis 通過@ResponseBody返回結(jié)果中文亂碼的解決方法
下面小編就為大家分享一篇spring+mybatis 通過@ResponseBody返回結(jié)果中文亂碼的解決方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2017-12-12Java實(shí)現(xiàn)Map集合二級(jí)聯(lián)動(dòng)示例
Java實(shí)現(xiàn)Map集合二級(jí)聯(lián)動(dòng)示例,需要的朋友可以參考下2014-03-03