1. 程式人生 > >使程式在Linux下後臺執行 (關掉終端繼續讓程式執行的方法)

使程式在Linux下後臺執行 (關掉終端繼續讓程式執行的方法)

一、為什麼要使程式在後臺執行

我們計算的程式都是週期很長的,通常要幾個小時甚至一個星期。我們用的環境是用putty遠端連線到日本Linux伺服器。所以使程式在後臺跑有以下三個好處:

1:我們這邊是否關機不影響日本那邊的程式執行。(不會像以前那樣,我們這網路一斷開,或一關機,程式就斷掉或找不到資料,跑了幾天的程式只能重頭再來,很是煩惱)

2:不影響計算效率

2:讓程式在後臺跑後,不會佔據終端,我們可以用終端做別的事情。

二、怎麼樣使程式在後臺執行

方法有很多,這裡主要列舉兩種。假如我們有程式pso.cpp,通過編譯後產生可執行檔案pso,我們要使pso在linux伺服器後臺執行。當客戶端關機後重新登入伺服器後繼續檢視本來在終端輸出的執行結果。(假設操作都在當前目錄下)

方法1在終端輸入命令:

 # ./pso > pso.file 2>&1 & 

解釋:將pso直接放在後臺執行,並把終端輸出存放在當前目錄下的pso.file檔案中。

      當客戶端關機後重新登陸伺服器後,直接檢視pso.file檔案就可看執行結果(命

令:#cat pso.file )。

  

方法2在終端輸入命令:

# nohup  ./pso > pso.file 2>&1 &

解釋:nohup就是不掛起的意思,將pso直接放在後臺執行,並把終端輸出存放在當前

目錄下的pso.file檔案中。當客戶端關機後重新登陸伺服器後,直接檢視pso.file

檔案就可看執行結果(命令:#cat pso.file )。

三、常用任務管理命令

# jobs      //檢視任務,返回任務編號n和程序號

# bg  %n   //將編號為n的任務轉後臺執行

# fg  %n   //將編號為n的任務轉前臺執行

# ctrl+z    //掛起當前任務

# ctrl+c    //結束當前任務

 

注:如果要使在前天執行任務放到後臺執行,則先要用ctrl+z掛起該任務,然後用bg使之後臺執行。

 

附:

  

在Linux中,如果要讓程序在後臺執行,一般情況下,我們在命令後面加上&即可,實際上,這樣是將命令放入到一個作業佇列中了:

$ ./test.sh & 
[1] 17208  
$ jobs -l 
[1]+ 17208 Running                 ./test.sh & 

對於已經在前臺執行的命令,也可以重新放到後臺執行,首先按ctrl+z暫停已經執行的程序,然後使用bg命令將停止的作業放到後臺執行:

$ ./test.sh 
[1]+  Stopped                 ./test.sh  
$ bg %1 [1]+ ./test.sh &  
$ jobs -l 
[1]+ 22794 Running                 ./test.sh & 

 

但是如上方到後臺執行的程序,其父程序還是當前終端shell的程序,而一旦父程序退出,則會發送hangup訊號給所有子程序,子程序收到hangup以後也會退出。如果我們要在退出shell的時候繼續執行程序,則需要使用nohup忽略hangup訊號,或者setsid將將父程序設為init程序(程序號為1)

$ echo $$ 
21734  
$ nohup ./test.sh & 
[1] 29016  
$ ps -ef | grep test 
515      29710 21734  0 11:47 pts/12   00:00:00 /bin/sh ./test.sh 
515      29713 21734  0 11:47 pts/12   00:00:00 grep test 
$ setsid ./test.sh & 
[1] 409  
$ ps -ef | grep test 
515        410     1  0 11:49 ?        00:00:00 /bin/sh ./test.sh 
515        413 21734  0 11:49 pts/12   00:00:00 grep test 

上面的試驗演示了使用nohup/setsid加上&使程序在後臺執行,同時不受當前shell退出的影響。那麼對於已經在後臺執行的程序,該怎麼辦呢?可以使用disown命令:

$ ./test.sh & 
[1] 2539  
$ jobs -l 
[1]+  2539 Running                 ./test.sh &  
$ disown -h %1  
$ ps -ef | grep test 
515        410     1  0 11:49 ?        00:00:00 /bin/sh ./test.sh 
515       2542 21734  0 11:52 pts/12   00:00:00 grep test 

另外還有一種方法,即使將程序在一個subshell中執行,其實這和setsid異曲同工。方法很簡單,將命令用括號() 括起來即可:

$ (./test.sh &)  
$ ps -ef | grep test 
515        410     1  0 11:49 ?        00:00:00 /bin/sh ./test.sh 
515      12483 21734  0 11:59 pts/12   00:00:00 grep test 

注:本文試驗環境為Red Hat Enterprise Linux AS release 4 (Nahant Update 5),shell為/bin/bash,不同的OS和shell可能命令有些不一樣。例如AIX的ksh,沒有disown,但是可以使用nohup -p PID來獲得disown同樣的效果。

還有一種更加強大的方式是使用screen,首先建立一個斷開模式的虛擬終端,然後用-r選項重新連線這個虛擬終端,在其中執行的任何命令,都能達到nohup的效果,這在有多個命令需要在後臺連續執行的時候比較方便:

$ screen -dmS screen_test  
$ screen -list There is a screen on:
              27963.screen_test       (Detached) 1 Socket in /tmp/uscreens/S-jiangfeng.  
$ screen -r screen_test