天天看點

java 版本的redis-stat不能運作在背景和daemon

在使用java版本的redis-stat時,在putty中打開redis-stat後,如果退出putty則redis-stat不能工作。而且以背景方式打開redis-stat也不行(後面加&).使用nohup還是不行,不過最後使用screen則可以,安裝screen的方法是yum install screen。現在還不知道原因,下來研究。

原因如下:

讓java程式在背景一直執行(例如putty關閉後背景程式繼續運作)

在linux系統下如何使一個程式在背景一直執行呢?很多人使用&結尾的指令形式,但是如果沒有守護程序的話,即使這樣,在終端關閉時程式也會終止運作。這時候就得使用nohup指令了。

     例如,如果在終端中執行java -jar xxx.jar&指令,當終端關閉時,xxx.jar也會同時結束運作,但是如果執行nohup java -jar xxx.jar&指令,則程式會在背景一直運作,值得注意的是,此時程式控制台輸出會被轉移到nohup.out檔案中。

附:nohup指令參考 nohup 指令   

用途:不挂斷地運作指令。   

文法:nohup Command [ Arg ... ] [ & ]   

描述:nohup 指令運作由 Command 參數和任何相關的 Arg 參數指定的指令,忽略所有挂斷(SIGHUP)信号。在登出後使用 nohup 指令運作背景中的程式。要運作背景中的 nohup 指令,添加 & ( 表示"and"的符号)到指令的尾部。   

     無論是否将 nohup 指令的輸出重定向到終端,輸出都将附加到目前目錄的 nohup.out 檔案中。如果目前目錄的 nohup.out 檔案不可寫,輸出重定向到 $HOME/nohup.out 檔案中。如果沒有檔案能建立或打開以用于追加,那麼 Command 參數指定的指令不可調用。如果标準錯誤是一個終端,那麼把指定的指令寫給标準錯誤的所有輸出作為标準輸出重定向到相同的檔案描述符。

如需要了解視窗關閉導緻運作程序關閉的原因:請參看下面的具體分析

--------------------------------------------------------------------------------

你是不是經常需要遠端登入到Linux伺服器?你是不是經常為一些長時間運作的任務頭疼?還在用 nohup 嗎?那麼來看看 screen 吧,它會給你一個驚喜! 你是不是經常需要 SSH 或者 telent 遠端登入到 Linux 伺服器?你是不是經常為一些長時間運作的任務而頭疼,比如系統備份、ftp 傳輸等等。通常情況下我們都是為每一個這樣的任務開一個遠端終端視窗,因為他們執行的時間太長了。必須等待它執行完畢,在此期間可不能關掉視窗或者斷開連接配接,否則這個任務就會被殺掉,一切半途而廢了。

【原因所在】

元兇:SIGHUP 信号 

讓我們來看看為什麼關掉視窗/斷開連接配接會使得正在運作的程式死掉。 

在Linux/Unix中,有這樣幾個概念:

?程序組(process group):一個或多個程序的集合,每一個程序組有唯一一個程序組ID,即程序組長程序的ID。

?會話期(session):一個或多個程序組的集合,有唯一一個會話期首程序(session leader)。會話期ID為首程序的ID。 

?會話期可以有一個單獨的控制終端(controlling terminal)。與控制終端連接配接的會話期首程序叫做控制程序(controlling process)。目前與終端互動的程序稱為前台程序組。其餘程序組稱為背景程序組。 

根據POSIX.1定義: 

挂斷信号(SIGHUP)預設的動作是終止程式。 

當終端接口檢測到網絡連接配接斷開,将挂斷信号發送給控制程序(會話期首程序)。 

如果會話期首程序終止,則該信号發送到該會話期前台程序組。

一個程序退出導緻一個孤兒程序組中産生時,如果任意一個孤兒程序組程序處于STOP狀态,發送SIGHUP和SIGCONT信号到該程序組中所有程序。 

是以當網絡斷開或終端視窗關閉後,控制程序收到SIGHUP信号退出,會導緻該會話期内其他程序退出。 

我們來看一個例子。打開兩個SSH終端視窗,在其中一個運作top指令。 

在另一個終端視窗,找到top的程序ID為5180,其父程序ID為5128,即登入shell。 

使用pstree指令可以更清楚地看到這個關系: 

使用ps-xj指令可以看到,登入shell(PID 5128)和top在同一個會話期,shell為會話期首程序,所在程序組PGID為5128,top所在程序組PGID為5180,為前台程序組。 

關閉第一個SSH視窗,在另一個視窗中可以看到top也被殺掉了。

(其實,對于windows控制台程式,也是類似的。在一個控制台視窗中運作一個非service程式,那麼控制台視窗自然是該視窗中運作程式的父程序,當父程序關閉時,預設由其建立的子程序會被kill掉。當然,應該也有機制建立類似unix的背景程序。。。)

【基礎回避方法】

    如果我們可以忽略SIGHUP信号,關掉視窗應該就不會影響程式的運作了。nohup指令可以達到這個目的,如果程式的标準輸出/标準錯誤是終端,nohup預設将其重定向到nohup.out檔案。值得注意的是nohup指令隻是使得程式忽略SIGHUP信号,還需要使用标記 & 把它放在背景運作。

【進階回避方法】

    雖然nohup很容易使用,但還是比較"簡陋"的,對于簡單的指令能夠應付過來,對于複雜的需要人機互動的任務就麻煩了。 

其實我們可以使用一個更為強大的實用程式screen。流行的Linux發行版(例如Red Hat Enterprise Linux 4)通常會自帶screen實用程式,如果沒有的話,可以從GNU screen的官方網站下載下傳。

開始使用Screen 

簡單來說,Screen是一個可以在多個程序之間多路複用一個實體終端的視窗管理器。Screen中有會話的概念,使用者可以在一個screen會話中建立多個screen視窗,在每一個screen視窗中就像操作一個真實的telnet/SSH連接配接視窗那樣。在screen中建立一個新的視窗有這樣幾種方式: 

1.直接在指令行鍵入screen指令 

Screen将建立一個執行shell的全屏視窗。你可以執行任意shell程式,就像在ssh視窗中那樣。在該視窗中鍵入exit退出該視窗,如果這是該screen會話的唯一視窗,該screen會話退出,否則screen自動切換到前一個視窗。 

2.Screen指令後跟你要執行的程式。 

Screen建立一個執行vi test.c的單視窗會話,退出vi将退出該視窗/會話。 

3.以上兩種方式都建立新的screen會話。我們還可以在一個已有screen會話中建立新的視窗。在目前screen視窗中鍵入C-a c,即Ctrl鍵+a鍵,之後再按下c鍵,screen 在該會話内生成一個新的視窗并切換到該視窗。 

screen還有更進階的功能。你可以不中斷screen視窗中程式的運作而暫時斷開(detach)screen會話,并在随後時間重新連接配接(attach)該會話,重新控制各視窗中運作的程式。例如,我們打開一個screen視窗編輯/tmp/abc檔案: 

之後我們想暫時退出做點别的事情,比如出去散散步,那麼在screen視窗鍵入C-a d,Screen會給出detached提示:

半個小時之後回來了,找到該screen會話: 

重新連接配接會話: 

看看出現什麼了,太棒了,一切都在。繼續幹吧。 

你可能注意到給screen發送指令使用了特殊的鍵組合C-a。這是因為我們在鍵盤上鍵入的資訊是直接發送給目前screen視窗,必須用其他方式向screen視窗管理器發出指令,預設情況下,screen接收以C-a開始的指令。這種指令形式在screen中叫做鍵綁定(key binding),C-a叫做指令字元(command character)。

以上原因引自:http://qsfwy.iteye.com/blog/1654411

繼續閱讀