大家都知道在Makefile可以調用shell腳本,但是Makefile和shell腳本是不同的。本文試着介紹一下Makefile和shell腳本的不同。
1、在Makefile中只能在target中調用Shell腳本,其他地方是不能輸出的。比如如下代碼就是沒有任何輸出:
VAR="Hello"
echo "$VAR"
all: .....
以上代碼任何時候都不會輸出,沒有在target內,如果上述代碼改為如下:
VAR="Hello"
all:
echo "$VAR" .....
以上代碼,在make all的時候將會執行echo命令。
2、在Makefile中執行shell命令,一行創建一個進程來執行。這也是為什么很多Makefile中有很多行的末尾都是“; \”,以此來保證代碼是一行而不是多行,這樣Makefile可以在一個進程中執行,例如:
SUBDIR=src example
all:
@for subdir in $(SUBDIR); \
do\
echo "building "; \
done
上述可以看出for循環中每行都是以”; \”結尾的。
3、Makefile中所有以$打頭的單詞都會被解釋成Makefile中的變量。如果你需要調用shell中的變量(或者正則表達式中錨定句位$),都需要加兩個$符號($$)。實例如下:
PATH="/data/"
all:
echo ${PATH}
echo $$PATH
例子中的第一個${PATH}引用的是Makefile中的變量,而不是shell中的PATH環境變量,后者引用的事Shell中的PATH環境變量。
以上三點的是Makefile調用shell應該注意的地方,寫Makefile一定要注意。
shell腳本條件判斷
UNIX Shell 編程中條件判斷是極為重要的,以下是常用的條件判斷:
-b file 若文件存在且是一個塊特殊文件,則為真
-c file 若文件存在且是一個字符特殊文件,則為真
-d file 若文件存在且是一個目錄,則為真
-e file 若文件存在,則為真
-f file 若文件存在且是一個規則文件,則為真
-g file 若文件存在且設置了SGID位的值,則為真
-h file 若文件存在且為一個符合鏈接,則為真
-k file 若文件存在且設置了”sticky”位的值
-p file 若文件存在且為一已命名管道,則為真
-r file 若文件存在且可讀,則為真
-s file 若文件存在且其大小大於零,則為真
-u file 若文件存在且設置了SUID位,則為真
-w file 若文件存在且可寫,則為真
-x file 若文件存在且可執行,則為真
-o file 若文件存在且被有效用戶ID所擁有,則為真
-z string 若string長度為0,則為真
-n string 若string長度不為0,則為真
string1 = string2 若兩個字符串相等,則為真
string1 != string2 若兩個字符串不相等,則為真
int1 -eq int2 若int1等於int2,則為真
int1 -ne int2 若int1不等於int2,則為真
int1 -lt int2 若int1小於int2,則為真
int1 -le int2 若int1小於等於int2,則為真
int1 -gt int2 若int1大於int2,則為真
int1 -ge int2 若int1大於等於int2,則為真
!expr 若expr為假則復合表達式為真。expr可以是任何有效的測試表達式
expr1 -a expr2 若expr1和expr2都為真則整式為真
expr1 -o expr2 若expr1和expr2有一個為真則整式為真
特殊變量
$0 正在被執行命令的名字。對於shell腳本而言,這是被激活命令的路徑
$n 該變量與腳本被激活時所帶的參數相對應。n是正整數,與參數位置相對應($1,$2…)
$# 提供腳本的參數號
$* 所有這些參數都被雙引號引住。若一個腳本接收兩個參數,$*等於$1$2
$@ 所有這些參數都分別被雙引號引住。若一個腳本接收到兩個參數,$@等價於$1$2
$? 前一個命令執行后的退出狀態
$$ 當前shell的進程號。對於shell腳本,這是其正在執行時的進程ID
$! 前一個后台命令的進程號
Shell函數
mylog() {
echo "[`date '+%F %T'`] [ $2.$1 ] $3" >> $4
}
s_time=`date +%s`
s_curren_dir=$(pwd)
s_logfile=$s_curren_dir/log/install.log.`date '+%Y%m%d'`
logID=001
mylog "$s_time" "$logID" "your messager." "$s_logfile" #shell腳本中函數的調用不能用(),並且參數使用空格分隔,不是符合一般的習慣
在這自定義函數我沒試過,操一段
其實定義起來很簡單, 沒有我想象中的 def, command, function 之類的關鍵字, 而是等同於變量定義, 下面是嘗試在實際項目中使用的函數定義:
[zrf@DMLinux pes]$ cat Makefile.common
# The following function make the input token uniq.
# the token may be not in the input order
# usage: uniq_result=$(call uniq, A B A B B C)
# expected result: uniq_result is A B C
uniq=$(shell echo $1 | sed 's/[ \t]\+/\n/g' | sed 's/^/ /' | sort -u | tr -d '\n')
# The following function link the -lnet* and -lwrap library statically while
# keep other libraries being link dynamically by wrapping the -lxxx option with
# -Wl,-Bstatic ... -Wl,-Bdynamic
link_some_lib_staticlly=$(shell echo $1 | sed 's/-l\(net[^ \t]*\|wrap\)\( -l\(net[^ \t]*\|wrap\)\)*/-Wl,-Bstatic & -Wl,-Bdynamic/g')
自定義函數的關鍵是一定要用=號, 而不是:=, :=與=的差別是:=是即時求值, =則是使用時才求值. 把:看作一個站着的小人, :=可以看作"立等可取"來助記.
命令定義中可以用$1, $2, $3來引用傳遞的參數, 注釋里已經有了自定義函數的使用
call是個特殊的make函數, 用於調用其它函數, 參數以逗號分隔.
借助這種方法, 可以讓那些頻繁而又重復的操作被封裝進一個函數里, 不需要到處復制, DRY遠處不在.
上面這兩個自定義函數的功能是:
讓 LDLIBS這樣的變量中內容精簡, 同時能靈活地指定某個庫是以靜態方式鏈接到最終可執行文件里, 而多數其它的常用庫, 如libc仍保持是動態鏈接.自:http://blog.chinaunix.net/u/8681/showart_2228861.html