0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

【Makefile】Makefile與shell命令的聯(lián)系

嵌入式物聯(lián)網(wǎng)開發(fā) ? 來源:嵌入式物聯(lián)網(wǎng)開發(fā) ? 作者:嵌入式物聯(lián)網(wǎng)開發(fā) ? 2022-07-11 09:06 ? 次閱讀

【Makefile】Makefile與shell命令為何總是藕斷絲連?

1 寫在前面

博主最近在項目實踐過程中,需要深度定制項目的Makefile,其中有些復(fù)雜的流程必須得借助shell腳本才能高效地完成,為此博主特意深入學(xué)習(xí)了在Makefile種調(diào)用shell命令的方法。

大家都知道shell命令是Linux的神器,學(xué)會靈活應(yīng)用它可以大大提升執(zhí)行的自動化效率。

Makefile也是GNU的另一大神器,大家要知道,大名鼎鼎的Linux內(nèi)核的編譯就是靠Makefile來支撐的,連這么龐大、復(fù)雜的系統(tǒng)編譯它都可以輕松駕馭,可想而知,它也是牛逼哄哄的存在。

我們試想一下,如果把Makefile和shell命令這兩大神器聯(lián)合在一起,會有怎樣的效果,我們一起來看下。

2 什么是Makefile?

這個問題,我在之前的博文中就有介紹過,感興趣的可以點過去看一看:【Linux + Makefile】十分鐘教你學(xué)會Makefile的FORCE。

下面我再總結(jié)下,到底什么是Makefile?

Makefile的基本形式如下所示:

TARGET : DEPENDENCES CMD

# TARGET:生成的目標(biāo),可以是一個文件,也可以是一個虛擬符號(非真實文件) # DEPENDENCES: 生成目標(biāo)的所有依賴,它是一個集合,可以只有一個文件,或者很多文件;也可以是虛擬符號 # CMD:把所有依賴生成目標(biāo)的執(zhí)行命令,其中CMD前面是一個TAB鍵,而不能是空格 Makefile的基本規(guī)則,歸納起來就是一句話:【當(dāng)一個TARGET (欲生成的目標(biāo))比它的任何一個DEPENDENCES(依賴的文件)舊時,這個TARGET就要重新生成】。

這段話讀起來很簡單,就是Makefile在執(zhí)行編譯的時候,會先去判斷TARGET和其依賴文件的時間差,如果TARGET的時間比較舊,意味著依賴文件已經(jīng)更新了,那么TARGET就要重新生成;反之,如果TARGET的時間比較新,意味著在TARGET生成后,依賴文件是沒有改變過的,自然就不用重新去生成TARGET。

說白了,就是搞清楚【TARGET 和 DEPENDENCES 的時間關(guān)系】,弄明白了這個,你就可以讀懂Makefile了。

3 如何調(diào)試Makefile?

以下Makefile給大家演示下如何在Makefile中寫注釋以及如何調(diào)試Makefile的變量。

# This is an annotation for this Makefile.

# Call shell cmd in these ways ...
CUR_PATH = `pwd`
LS_FILES = "`ls`"
LS_TEST_FILES = "$(shell ls test*)"
DEBUG_VARIABLE = "debug"

# Debug your Makefile variables ...
$(info "This is a info msg ...")
$(warning "This is a warning msg ... DEBUG_VARIABLE=$(DEBUG_VARIABLE)")
$(error "This is a error msg ... It will make this Makefile to exist beacuse of an 'error' accur.")

# Begin you targets and dependences

info:
    @echo "CUR_PATH = $(CUR_PATH)"
    @echo "LS_FILES = $(LS_FILES)"
    @echo "LS_TEST_FILES = $(LS_TEST_FILES)"

我們執(zhí)行下make info可以看下,輸出信息

img

分別對應(yīng)info,warning,error;這三個函數(shù)是Makefile中內(nèi)置的,在任意Makefile中都可以使用。但是需要注意的,這三個函數(shù)可以在Makefile的任意位置調(diào)用,但是必須符合語法規(guī)則,比如在CMD部分(見第一章節(jié)的引用內(nèi)容)調(diào)用時,必須時TAB開頭。

我們試下這個的效果:

imgimg

4 Makefile定義變量的時候調(diào)用shell命令

還是拿上面的Makefile來分析,我們一開始定義了幾個變量:

CUR_PATH = `pwd`
LS_FILES = "`ls`"
LS_TEST_FILES = "$(shell ls test*)"

其中CURPATH是保存當(dāng)前Makefile的路徑,我們都知道shell命令有個pwd可以獲取到當(dāng)前目錄的路徑,所以 CURPATH 變量正是利用了這個命令來取得當(dāng)前路徑。其中,值得注意的是 表示的是獲取pwd命令的返回值,這個,不是分號 ' ',而是鍵盤左上角數(shù)字1左邊的那個鍵值,新手一般非常容易搞錯這一點。

第二個變量LS_FILES的定義,使用同樣的方式調(diào)用了 ls 命令,但是它在返回值的左右,加了 “ ”;這樣做的主要目的是,ls的返回值通常是一個字符串,中間每個文件名用空格隔開的;如果我們想把這一系列的字符串連起來而不是分散的,我們就必須使用 " "把它們包起來,這樣Makefile變量就會把它當(dāng)作一個字符串整體來處理。

第三個變量LSTESTFILES,它就換了一種方式來調(diào)用shell命令,它的格式是 $(shell xxx),其中xxx就是一個可以在shell環(huán)境下執(zhí)行的shell命令,比如本例子中,xxx就是 ls test*。

兩者方法沒有本質(zhì)的區(qū)別,結(jié)合自己的實際情況來使用即可,從易讀性來說,推薦使用$(shell xxx)的調(diào)用方式。

5 Makefile在執(zhí)行命令的時候調(diào)用shell命令

以下面的Makefile為例子:

# This is an annotation for this Makefile.

shell_cmd:
    @ls test*
    @pwd
    @echo "shell_cmd test ..."

執(zhí)行make shell_cmd,我們可以看到輸出:

img

只要是shell執(zhí)行的命令,都可以運行,這就方便我們做一些自動化操作,比如創(chuàng)建目錄、刪除文件、下載文件、上傳文件等等。

6 Makefile以函數(shù)的形式調(diào)用shell命令

請看下面的Makefile內(nèi)容

# This is an annotation for this Makefile.

DEBUG_VARIABLE = "debug"

# This is a shell function.
define function_test_shell
    param1=$1;\
    local_variable=12345678;\
    debug_variable=$(DEBUG_VARIABLE);\
    echo "param1=$$param1";\
    echo "local_variable=$$local_variable";\
    echo "debug_variable=$$debug_variable";\
    i=0;\
    cnt=10;\
    for i in `seq $$cnt`; do\
        echo "i=$$i";\
    done;\
    ls -al;\
    ls -al test*
endef

# Begin you targets and dependences

shell_function:
    @$(call function_test_shell,$(DEBUG_VARIABLE))

我們執(zhí)行make shell_function可以看到輸出:

pYYBAGLK8s2ACqvEAABZItmdNSQ889.png

對應(yīng)就是shell函數(shù)里面的操作。

在Makefile調(diào)用shell函數(shù)的寫法上,有幾點特別需要注意:

1)Makefile定義函數(shù)是使用define 和 endef,務(wù)必配對使用;函數(shù)里面的實現(xiàn)可以調(diào)用shell命令,也可以調(diào)用Makefile內(nèi)置的函數(shù),也可以調(diào)用自定義的Makefile函數(shù);

2)Makefile函數(shù)的調(diào)用方式是$(call function,xx,yy,zz),其中function是函數(shù)名,也就是define后面的名字,xx/yy/zz分別是函數(shù)的三個入?yún)?;?dāng)然有幾個參數(shù)就填幾個,每一個使用 , 隔開;

3) 在函數(shù)里面使用$1 $2 $3獲取輸入的參數(shù),即$+數(shù)字的方式獲取入?yún)ⅲ?/p>

4)如果函數(shù)里面使用的是shell組合命令,請務(wù)必在每一條shell命令后面加上 ; 符號,并且每一行加上 \ 連起來;這樣對于shell解析器而言,它會認(rèn)為是一條長長的組合輸入命令,而不報語法錯誤;

5)函數(shù)內(nèi)部使用shell的變量,比如是變量a,正確的寫法是$$a;而不是常規(guī)的shell寫法$a,也不是Makefile獲取變量的方法$(a)或${a};這一點特別需要注意。

6)如果函數(shù)內(nèi)部訪問的是Makefile定義的變量,比如示例中的訪問DEBUGVARIABLE變量,還是采用$(DEBUGVARIABLE)的方式。


好了,本期分享就到這里,有疑問的話,歡迎聯(lián)系我。

7 更多分享

歡迎關(guān)注我的github倉庫01workstation,日常分享一些開發(fā)筆記和項目實戰(zhàn),歡迎指正問題。

同時也非常歡迎關(guān)注我的CSDN主頁和專欄:

【CSDN主頁-架構(gòu)師李肯】

RT-Thread主頁-架構(gòu)師李肯】

【C/C++語言編程專欄】

【GCC專欄】

【信息安全專欄】

【RT-Thread開發(fā)筆記】

freeRTOS開發(fā)筆記】

【BLE藍(lán)牙開發(fā)筆記】

ARM開發(fā)筆記】

RISC-V開發(fā)筆記】

有問題的話,可以跟我討論,知無不答,謝謝大家。

審核編輯:湯梓紅

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11304

    瀏覽量

    209541
  • Makefile
    +關(guān)注

    關(guān)注

    1

    文章

    125

    瀏覽量

    19185
  • Shell
    +關(guān)注

    關(guān)注

    1

    文章

    365

    瀏覽量

    23379
收藏 人收藏

    評論

    相關(guān)推薦

    linux系統(tǒng)中Makefile的使用方法

    Makefile是一種編譯控制文件,廣泛用于項目的自動化構(gòu)建。它定義了一系列的規(guī)則來指導(dǎo)構(gòu)建的過程。通過Makefile,開發(fā)者可以輕松管理大型項目的編譯鏈接、清理等任務(wù)。本文將從Makefile
    的頭像 發(fā)表于 05-11 08:49 ?673次閱讀

    Linux中makefileshell的區(qū)別

    :*** missing separator.Stop. 5、在Makefile中執(zhí)行shell命令,一行創(chuàng)建一個進(jìn)程來執(zhí)行。更多學(xué)習(xí)信息交流盧工3311615775,這也是為什么很多Make
    發(fā)表于 09-22 14:29

    無法在2015.3 sdk中運行makefile

    17-259] Unknown Tcl命令'make -f Makefile'發(fā)送命令到OS shell執(zhí)行.make.EXE:沒有什么可以為'all'做。以上來自于谷歌翻譯以下為原
    發(fā)表于 04-17 10:04

    Makefile編寫

    )shell函數(shù) 執(zhí)行shell命令例如:$(shell ls -d */)當(dāng)我們學(xué)會了Makefile編寫就可以讓其為我們所用了,有什么疑
    發(fā)表于 11-02 15:59

    C語言入門教程-makefile

    makefile 重復(fù)鍵入上一節(jié)的gcc命令很是麻煩,尤其是當(dāng)您要不斷修改的代碼,且使用了好幾個函數(shù)庫的時候。make工具解決了這個問題。您可以用下面的makefile代替前面的一串編
    發(fā)表于 07-29 11:20 ?901次閱讀

    駕馭Makefile

    駕馭Makefile在網(wǎng)上你能找到很多關(guān)于Makefile的學(xué)習(xí)資料,但絕大部分給你的只是一個知識點,與將Makefile運用到項目中(尤其是大型項目)的差距非常的大。因為,將Makefile
    發(fā)表于 01-05 17:05 ?9次下載

    linux makefile教程

     什么是makefile?或許很多Winodws的程序員都不知道這個東西,因為那些Windows的IDE都為你做了這個工作,但我覺得要作一個好的和professional的程序員,makefile
    發(fā)表于 11-12 09:11 ?5328次閱讀

    makefile的基本語法

     在Makefile中,最重要的三個概念是:目標(biāo)(target)、依賴關(guān)系(dependency)和命令(command)。目標(biāo)是指要干什么,即運行make后生成什么;依賴是指明目標(biāo)所依賴的其他目標(biāo);命令則告訴make如何生成目
    發(fā)表于 11-12 10:15 ?1.1w次閱讀

    嵌入式系統(tǒng)Kconfig .config Makefile聯(lián)系和區(qū)別

    2.6內(nèi)核的Makefile分為5個組成部分: 1. 最頂層的Makefile :讀取 .config文件的內(nèi)容,并總體上負(fù)責(zé)build內(nèi)核和模塊。 2. 內(nèi)核的.config配置文件 3. 在
    的頭像 發(fā)表于 08-01 16:20 ?5627次閱讀

    Makefile是什么?Makefile工作原理是怎樣的?Makefile經(jīng)典教程免費下載

    Makefile的重要性 會不會寫makefile,從一個側(cè)面說明了一個人是否具備完成大型工程的能力 makefile帶來的好處就是——“自動化編譯”,一旦寫好,只需要一個make命令
    發(fā)表于 09-12 17:19 ?0次下載
    <b class='flag-5'>Makefile</b>是什么?<b class='flag-5'>Makefile</b>工作原理是怎樣的?<b class='flag-5'>Makefile</b>經(jīng)典教程免費下載

    嵌入式中的Makefile應(yīng)用

    文章目錄一.Makefile 引入二. Makefile語法1.通配符2.假象目標(biāo)3.變量三.Makefile函數(shù)四.實例本文主要總結(jié)一下嵌入式開發(fā)中的Makefile,一般項目中都需
    發(fā)表于 11-03 17:06 ?11次下載
    嵌入式中的<b class='flag-5'>Makefile</b>應(yīng)用

    Makefile】簡單實用的Makefile模板來了

    【Linux + Makefile】簡單實用的Makefile模板來了
    的頭像 發(fā)表于 08-31 12:46 ?1906次閱讀
    【<b class='flag-5'>Makefile</b>】簡單實用的<b class='flag-5'>Makefile</b>模板來了

    Linux開發(fā)_Makefile規(guī)則與Shell腳本語言

    介紹Linux下Makefile編程知識點,Shell腳本知識點。
    的頭像 發(fā)表于 09-17 15:40 ?1290次閱讀

    什么是Makefile?

    如果您有多個 c、c++ 和其他語言的文件,并且想通過終端命令編譯它們,我們該如何編譯他們呢?為了解決這類問題,Makefile就出現(xiàn)了。
    的頭像 發(fā)表于 02-17 10:41 ?4583次閱讀
    什么是<b class='flag-5'>Makefile</b>?

    Makefile可以做什么?Makefile的基本格式

    Makefile可以根據(jù)指定的依賴規(guī)則和文件是否有修改來執(zhí)行命令。常用來編譯軟件源代碼,只需要重新編譯修改過的文件,使得編譯速度大大加快。
    的頭像 發(fā)表于 01-25 11:18 ?707次閱讀