99精品伊人亚洲|最近国产中文炮友|九草在线视频支援|AV网站大全最新|美女黄片免费观看|国产精品资源视频|精彩无码视频一区|91大神在线后入|伊人终合在线播放|久草综合久久中文

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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

【Makefile】C文件包含的頭文件修改,但不重新編譯?

嵌入式物聯網開發(fā) ? 來源:嵌入式物聯網開發(fā) ? 作者:嵌入式物聯網開發(fā) ? 2022-09-08 08:53 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

在上一篇 《【Linux + Makefile】十分鐘教你學會Makefile的FORCE》文章的最后,筆者就FORCE的用法在一個示例工程中使用,提出了一個問題:為何build_info.h每次都是新生成的(有修改過),而main.c又是有include “build-info.h”,但main.c卻不是每次都重新編譯呢?這個到底是不是違反了Makefile的基本規(guī)則呢?本文將給你答案,通過閱讀本文,你將了解到以下內容:

  • 如何保證在C文件中包含的頭文件修改了的時候,C文件每次都會被重新編譯?

為了更好地展示上訴描述的問題,我們將之前的示例工程稍微復雜化一點點:

整個工程有3個.c文件,a.c/b.c/main.c,其中main.c會調用a.c/b.c中的兩個接口,同時main.c會include頭文件build_info.h;這個build_info.h每次編譯都會重新生成,按照我們之前的寫法,我們Makefile可能就是這樣:

SHELL           = /bin/bash #指定shell使用/bin/bash,否則echo -e可能會出問題
ECHO            = echo
BIN             = test
BUILG_INFO_H    = build_info.h
SRC-C-y         += a.c
SRC-C-y         += b.c
SRC-C-y         += main.c
SRC-O           = $(patsubst %.c, $(O)%.o, $(SRC-C-y))

all: gen_build_info $(BIN)

clean: 
    rm -rf $(SRC-O) $(BIN) $(BUILG_INFO_H)

$(BIN) : $(SRC-O)
    gcc -o $(O)"$@" $(SRC-O)
	
%.o : %.c
    gcc -c "$<" -o "$@"
	
gen_build_info: $(BUILG_INFO_H)

$(BUILG_INFO_H): FORCE     #強制生成build_info.h
    @$(RM) $@
    @$(ECHO) '  GEN     $@'
    @$(ECHO) -e   " #ifndef __BUILD_INFO_H__\n"\
				"#define __BUILD_INFO_H__\n"\
				"#define APP_TIME        	\"`date "+%Y-%m-%d %H:%M:%S"`\"\n"\
				"#endif"  > $@

FORCE:
.PHONY: FORCE
poYBAGDYdXCAWkKMAAAAK8RNs4s030.png

執(zhí)行make,我們會發(fā)現,跟我們的預期不一樣:它雖然會每次都生成build_info.h,但是main.c包含了build_info.h卻不會每次都重新編譯。這個問題發(fā)生的原因,我們來分析下:

在我們的Makefile規(guī)則中,main.o只依賴于main.c (Makefile 第18-19行),而在第二次執(zhí)行make的時候,main.c顯然并沒有被修改,所以main.o不會重新生成,自然可執(zhí)行文件就不會重新生成。這里的問題根源在于,main.c它是依賴于build_info.h的,而這個依賴關系并沒有體現在Makefile中,所以整個編譯流程達不到我們的預期想法。我們嘗試下,將main.c的依賴頭文件也寫入到Makefile中,怎么實現呢?

恰好,GCC給了我們強大的支持,它有個非常有用的選項 -MD -MF,它可以在生成一個.o的同時也生成它的依賴文件列表,修改后的Makefile如下所示:

SHELL           = /bin/bash #指定shell使用/bin/bash,否則echo -e可能會出問題
ECHO            = echo
BIN             = test
BUILG_INFO_H    = build_info.h
SRC-C-y         += a.c
SRC-C-y         += b.c
SRC-C-y         += main.c
SRC-O           = $(patsubst %.c, $(O)%.o, $(SRC-C-y))
SRC-C-DEPS      = $(patsubst %.c, $(O).%.o.d, $(SRC-C-y))  ## 由 a.c ==> .a.o.d

all: gen_build_info $(BIN)

clean: 
    rm -rf $(SRC-O) $(BIN) $(BUILG_INFO_H) $(SRC-C-DEPS)

$(BIN) : $(SRC-O)
    gcc -o $(O)"$@" $(SRC-O)
	
%.o : %.c
#	生成xxx.o的時候,同時生成它的依賴列表,放在文件.xxx.o.d中
	gcc -c "$<" -o "$@" -MD -MF "$(dir $@).$(notdir $@).d" -MT "$@"
	
gen_build_info: $(BUILG_INFO_H)

$(BUILG_INFO_H): FORCE     #強制生成build_info.h
    @$(RM) $@
    @$(ECHO) '  GEN     $@'
    @$(ECHO) -e   " #ifndef __BUILD_INFO_H__\n"\
				"#define __BUILD_INFO_H__\n"\
				"#define APP_TIME        	\"`date "+%Y-%m-%d %H:%M:%S"`\"\n"\
				"#endif"  > $@

FORCE:
.PHONY: FORCE

# 在Makefile末尾強制包含這些依賴文件
-include $(SRC-C-DEPS)

測試結果如下所示:

再次執(zhí)行make,多試幾次,一樣的結果。

由上可知,經過改造后的Makefile是實現了我們的需求,每次build_info.h重新生成,導致main.c包含了build_info.h也會重新編譯,而a.c和b.c沒有被修改,所以在未執(zhí)行make clean的情況下,a.c和b.c是不會被重新編譯的,每次都是僅僅main.c被再次編譯,從而重新生成新的test可執(zhí)行文件。這樣就是已經達到了【當C文件包含的頭文件修改了的時候,C文件必須重新編譯】的目的。


以上就是關于Makefile的高階用法,基本滿足了我們日常工程實踐的需求。如果你對該Makefile有疑問,歡迎在評論席提出你的疑問,博主很樂意為你解答。


延伸閱讀:

【Linux + Makefile】十分鐘教你學會Makefile的FORCE

?審核編輯:湯梓紅

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

    關注

    87

    文章

    11511

    瀏覽量

    213838
  • Makefile
    +關注

    關注

    1

    文章

    125

    瀏覽量

    19735
  • C文件
    +關注

    關注

    0

    文章

    12

    瀏覽量

    3099
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    cypress3014在頭文件修改編譯后,時間不變,這樣頭文件的程序有被執(zhí)行嗎?

    你好,請問我在頭文件修改編譯后,時間不變,這樣頭文件的程序有被執(zhí)行嗎,怎么解決這個問題
    發(fā)表于 05-14 06:50

    STM32使用ISp燒錄HEX文件運行,代碼重新編譯之后的產生的新hex文件選擇全片擦除就沒辦法正常運行,為什么?

    我有一套代碼需要使用ISp燒錄HEX文件運行,代碼有一個老的HEX,無論是全片擦除還是擦除重要部分再下載都沒有問題。但是代碼重新編譯之后的產生的新hex文件,如果選擇全片擦除,就沒辦法正常運行。只有
    發(fā)表于 03-10 07:42

    飛凌嵌入式ElfBoard ELF 1板卡-初識設備樹之Makefile修改

    不同而新增加了dts,則需要在這個Makefile的這個位置添加上對應的.dtb文件名參與編譯。ELF 1使用的設備樹命名為imx6ull-elf1-emmc.dts,是基于NXP官方evk板子的設備樹imx6ull-14x14
    發(fā)表于 01-10 09:23

    嵌入式學習-飛凌嵌入式ElfBoard ELF 1板卡-Linux內核移植之Makefile介紹

    文件的目標文件進行重新編譯更新,這就大大減少了編譯時間。打開源碼目錄可以看到,Makefile文件
    發(fā)表于 01-04 10:40

    飛凌嵌入式ElfBoard ELF 1板卡-Linux內核移植之Makefile介紹

    文件的目標文件進行重新編譯更新,這就大大減少了編譯時間。 打開源碼目錄可以看到,Makefile文件
    發(fā)表于 01-03 09:39

    C語言中的頭文件能不能重復包含

    "); return 0;} 我們先來編譯看下,好像沒有任何問題。 root@Turbo:~# gcc test.c -o test root@Turbo:~# 同一個頭文件,隨便包含
    的頭像 發(fā)表于 11-26 17:19 ?611次閱讀

    編譯器怎么處理同名頭文件

    C語言中的include很簡單,但不是你想象中的簡單。
    的頭像 發(fā)表于 11-05 16:51 ?853次閱讀
    <b class='flag-5'>編譯</b>器怎么處理同名<b class='flag-5'>頭文件</b>

    labview程序生成exe文件怎么還原

    在LabVIEW中,程序生成exe文件后,通常這個過程是不可逆的,即exe文件無法直接“還原”回原始的LabVIEW項目文件(.vi或.lvproj)。exe文件是一個
    的頭像 發(fā)表于 09-04 17:12 ?3321次閱讀

    Keil工程下改動.c文件編譯的.o文件不更新如何解決?

    在維護公司項目的時候,打開工程下一個C文件在里面添加了一些代碼,然后點擊重新編譯后發(fā)現它的二進制O文件沒有同步更新,而其他C
    發(fā)表于 09-04 10:35

    單片機hex文件編譯C語言的過程

    使用C語言編寫,然后編譯成機器碼并燒錄到單片機的存儲器中。 Hex文件是一種用于存儲單片機程序的文件格式,它包含了程序的機器碼和一些附加信息
    的頭像 發(fā)表于 09-02 10:49 ?6146次閱讀

    hex文件如何查看原c語言代碼

    直接將 .hex 文件轉換回原始的 C 語言代碼是不可能的,因為 .hex 文件是二進制文件,它包含了單片機程序
    的頭像 發(fā)表于 09-02 10:37 ?4958次閱讀

    linux驅動程序的編譯方法有哪兩種

    Collection)或其他C/C++編譯器來編譯源代碼文件。這種方法較為原始,需要開發(fā)者手動指定編譯
    的頭像 發(fā)表于 08-30 14:39 ?1536次閱讀

    可重復頭文件的固定結構

    年輕人,你可曾記得,在修習C語言的時候,見過這樣的字句:在創(chuàng)建頭文件的時候,一定要加入保護宏。
    的頭像 發(fā)表于 08-29 10:23 ?630次閱讀
    可重復<b class='flag-5'>頭文件</b>的固定結構

    關于Makefile自動生成-autotools的使用

    在Linux應用開發(fā)中,編寫Makefile是一項必備技能,因為它定義了工程中所有文件編譯順序、規(guī)則和依賴關系,決定了哪些文件需要編譯以及
    的頭像 發(fā)表于 07-25 15:50 ?1829次閱讀
    關于<b class='flag-5'>Makefile</b>自動生成-autotools的使用

    如何修改buildroot和debian文件系統(tǒng)

    本文檔主要介紹在沒有編譯環(huán)境的情況下,如何修改buildroot和debian文件系統(tǒng)方法,如在buildroot文件系統(tǒng)中添加文件、
    的頭像 發(fā)表于 07-22 17:46 ?894次閱讀
    如何<b class='flag-5'>修改</b>buildroot和debian<b class='flag-5'>文件</b>系統(tǒng)