在之前一次編寫makfile時候,有看到相關的makefile中使用$$來引用變量,而且嘗試后發現$$使用居然和${}有類似的功能。當時也沒具體追究相關的用法,當然剛才所說的都是錯誤的觀念
$$:在makefile中會被替換成一個$。
??? 相關資料是這么描述的:makefile中變量在聲明時需要賦予初值,在使用時,需要在使用的變量前加"$"符號,但是最好使用"()"或者"{}"將變量包括起來。如果想使用真實的"$"符號,需要用"$$"來表示
在makefile中使用shell:
1. 使用makefile中的$(shell ...) eg:CUR_DIR=$(shell pwd)
2. 直接使用shell。在此有幾點需要注意的:
- shell命令必須是在規則里面(可以參考以下例1)
- shell命令在makefile調用時候每行shell都是一個單獨的進程。上一行定義的變量在下一行是無效的(參考例2)
- makefile調用shell時候(其實在執行時候都會有這個動作,不光光是shell),會對變量進行替換:如果是makefile中定義的變量(使用$()來引用的)會被替換成變量的值;如果是$$符號,會替換成$。然后傳給shell來執行
--------------------------------------------------------------------------------------
--------------------------------------------------------------------------------------
以下內容轉自:Makefile與Shell的問題
大概只要知道Makefile的人,都知道Makefile可以調用Shell腳本。但是在實際使用時,并不那么簡單,一些模棱兩可的地方可能會讓你抓狂。你若不信,可以先看幾個例子,想象一下這些這些例子會打印什么內容,記下你想象的結果,然后在計算機上運行這些例子,對照看一下。
?
示例一: |
if [ "$(BUILD)" = "debug" ]; then??echo "build debug"; else echo "build release"; fi all: ????echo "done" |
示例二: |
all: ????@CC=arm-linux-gcc ????@echo $(CC) |
示例三: |
CC=arm-linux-gcc all: ????@echo $(CC) |
示例四: |
SUBDIR=src example all: ????@for subdir in $(SUBDIR); / ????do/ ????????echo "building " $(subdir); / ????done |
?
?
說明:
1.?????????Shell腳本在target里才有效,其它地方都被忽略掉了。所以示例一中,”build debug”之類的字符串根本打印不出來。示例一的正確寫法是:
示例一: |
all: ????if [ "$(BUILD)" = "debug" ]; then??echo "build debug"; else echo "build release"; fi ????echo "done" |
?
2.?????????make把每一行Shell腳本當作一個獨立的單元,它們在單獨的進程中運行。示例二中,兩行Shell腳本在兩個莫不相干的進程里運行,第一個進程把CC設置為arm-linux-gcc,第二個進程是不知道的,所以打印的結果自然不是arm-linux-gcc了。示例二的正確寫法是:
示例二: |
all: ????@CC=arm-linux-gcc; echo $(CC) |
或者: |
all: @CC=arm-linux-gcc; / echo $(CC) |
?
3.?????????make在調用Shell之前先進行預處理,即展開所有Makefile的變量和函數。這些變量和函數都以$開頭。示例三中,Shell拿的腳本實際上是echo arm-linux-gcc,所以打印結果正確。
?
4.?????????make預處理時,所有以$開頭的,它都不會放過。要想引用Shell自己的變量,應該以$$開頭。另外要注意,Shell自己的變量是不需要括號的。示例四的正確寫法是:
示例四: |
SUBDIR=src example all: ????@for subdir in $(SUBDIR); / ????do/ ????????echo "building " $$subdir; / ????done |