1. 程式人生 > >Linux執行shell指令碼方式及區別&命令後臺執行

Linux執行shell指令碼方式及區別&命令後臺執行

假設shell指令碼檔案為hello.sh
放在/root目錄下。下面介紹幾種在終端執行shell指令碼的方法:

[[email protected] home]# cd /root/

[[email protected] ~]#vim hello.sh

#!  /bin/bash

cd /tmp

echo "hello guys!"

echo "welcome to my Blog:linuxboy.org!"

1.切換到shell指令碼所在的目錄,執行:

[[email protected] ~]# ./hello.sh

-bash: ./ hello.sh: 許可權不夠

2.以絕對路徑的方式執行:

[[email protected] ~]# /root/Desktop/hello.sh

-bash: /root/Desktop/ hello.sh: 許可權不夠

3.直接用bash或sh執行:

[[email protected] ~]# bash hello.sh

hello guys!

welcome to my Blog:linuxboy.org!

[[email protected] ~]# pwd

/root

[[email protected] ~]# sh hello.sh

hello guys!

welcome to my Blog:linuxboy.org!

[[email protected] ~]# pwd

/root

注意:用以上三種方法執行shell指令碼,現行的shell會開啟一個子shell環境,去執行shell指令碼,前兩種必須要有執行許可權才能夠執行

也可以讓shell指令碼在現行的shell中執行:

4.現行的shell中執行

[[email protected] ~]# . hello.sh

hello guys!

welcome to my Blog:linuxboy.org!

[[email protected] tmp]# pwd

/tmp

[[email protected] ~]# source hello.sh

hello guys!

welcome to my Blog:linuxboy.org!

[[email protected] tmp]# pwd

/tmp

對於第4種不會建立子程序,而是在父程序中直接執行

上面的差異是因為子程序不能改變父程序的執行環境,所以cd(內建命令,只有內建命令才可以改變shell 的執行環境)沒有成功,但是第4種沒有子程序,所以cd成功

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

   有兩種方式:

   1. command & : 後臺執行,你關掉終端會停止執行
   2. nohup command & : 後臺執行,你關掉終端也會繼續執行

一、 簡介
    Linux/Unix 區別於微軟平臺最大的優點就是真正的多使用者,多工。因此在任務管理上也有別具特色的管理思想。
我們知道,在 Windows 上面,我們要麼讓一個程式作為服務在後臺一直執行,要麼停止這個服務。而不能讓程式在前臺後臺之間切換。而 Linux 提供了 fg 和bg 命令,讓你輕鬆排程正在執行的任務。假設你發現前臺執行的一個程式需要很長的時間,但是需要幹其他的事情,你就可以用 Ctrl-Z ,掛起這個程式,然後可以看到系統提示:
[1]+ Stopped /root/bin/rsync.sh
然後我們可以把程式排程到後臺執行:(bg 後面的數字為作業號)
#bg 1
[1]+ /root/bin/rsync.sh &
用 jobs 命令檢視正在執行的任務:
#jobs
[1]+ Running /root/bin/rsync.sh &
如果想把它調回到前臺執行,可以用
#fg 1
/root/bin/rsync.sh
這樣,你在控制檯上就只能等待這個任務完成了。

& 將指令丟到後臺中去執行
[ctrl]+z 將前臺任務丟到後臺中暫停
jobs 檢視後臺的工作狀態
fg %jobnumber 將後臺的任務拿到前臺來處理
bg %jobnumber 將任務放到後臺中去處理
kill 管理後臺的任務

二、&

在Linux中,當在前臺執行某個作業時,終端被該作業佔據;而在後臺執行作業時,它不會佔據終端。可以使用&命令把作業放到後臺執行。實際上,這樣是將命令放入到一個作業佇列中了:

$ ./test.sh &
[1] 17208

$ jobs -l
[1]+ 17208 Running                 ./test.sh &
    在後臺執行作業時要當心:需要使用者互動的命令不要放在後臺執行,因為這樣你的機器就會在那裡傻等。不過,作業在後臺執行一樣會將結果輸出到螢幕上,干擾你的工作。如果放在後臺執行的作業會產生大量的輸出,最好使用下面的方法把它的輸出重定向到某個檔案中:
command >out.file 2>&1 &
在上面的例子中,2>&1表示所有的標準輸出和錯誤輸出都將被重定向到一個叫做out.file 的檔案中。 當你成功地提交程序以後,就會顯示出一個程序號,可以用它來監控該程序,或殺死它。
例:查詢名為“httpd.conf”的檔案,並把所有標準輸出和錯誤輸出重定向到find.dt的檔案中:
# find /etc/httpd/ -name "httpd.conf" -print >find.dt 2>&1 &
[2] 7832
成功提交該命令之後,系統給出了它的程序號7832。 對於已經在前臺執行的命令,也可以重新放到後臺執行,首先按ctrl+z暫停已經執行的程序,然後使用bg命令將停止的作業放到後臺執行,例如對正在前臺執行的tesh.sh使用ctrl+z掛起它:
$ ./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

來源:http://www.cnblogs.com/lwm-1988/archive/2011/08/20/2147299.html