一区二区三区三上|欧美在线视频五区|国产午夜无码在线观看视频|亚洲国产裸体网站|无码成年人影视|亚洲AV亚洲AV|成人开心激情五月|欧美性爱内射视频|超碰人人干人人上|一区二区无码三区亚洲人区久久精品

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

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

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

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

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

掃碼添加小助手

加入工程師交流群

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

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

為了更好地展示上訴描述的問(wèn)題,我們將之前的示例工程稍微復(fù)雜化一點(diǎn)點(diǎn):

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

SHELL           = /bin/bash #指定shell使用/bin/bash,否則echo -e可能會(huì)出問(wèn)題
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     #強(qiáng)制生成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,我們會(huì)發(fā)現(xiàn),跟我們的預(yù)期不一樣:它雖然會(huì)每次都生成build_info.h,但是main.c包含了build_info.h卻不會(huì)每次都重新編譯。這個(gè)問(wèn)題發(fā)生的原因,我們來(lái)分析下:

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

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

SHELL           = /bin/bash #指定shell使用/bin/bash,否則echo -e可能會(huì)出問(wèn)題
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的時(shí)候,同時(shí)生成它的依賴列表,放在文件.xxx.o.d中
	gcc -c "$<" -o "$@" -MD -MF "$(dir $@).$(notdir $@).d" -MT "$@"
	
gen_build_info: $(BUILG_INFO_H)

$(BUILG_INFO_H): FORCE     #強(qiáng)制生成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末尾強(qiáng)制包含這些依賴文件
-include $(SRC-C-DEPS)

測(cè)試結(jié)果如下所示:

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

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


以上就是關(guān)于Makefile的高階用法,基本滿足了我們?nèi)粘9こ虒?shí)踐的需求。如果你對(duì)該Makefile有疑問(wèn),歡迎在評(píng)論席提出你的疑問(wèn),博主很樂(lè)意為你解答。


延伸閱讀:

【Linux + Makefile】十分鐘教你學(xué)會(huì)Makefile的FORCE

?審核編輯:湯梓紅

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

    關(guān)注

    87

    文章

    11511

    瀏覽量

    213825
  • Makefile
    +關(guān)注

    關(guān)注

    1

    文章

    125

    瀏覽量

    19735
  • C文件
    +關(guān)注

    關(guān)注

    0

    文章

    12

    瀏覽量

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

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

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

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

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

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

    飛凌嵌入式ElfBoard ELF 1板卡-初識(shí)設(shè)備樹(shù)之Makefile修改

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

    嵌入式學(xué)習(xí)-飛凌嵌入式ElfBoard ELF 1板卡-Linux內(nèi)核移植之Makefile介紹

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

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

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

    C語(yǔ)言中的頭文件能不能重復(fù)包含

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

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

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

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

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

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

    在維護(hù)公司項(xiàng)目的時(shí)候,打開(kāi)工程下一個(gè)C文件在里面添加了一些代碼,然后點(diǎn)擊重新編譯后發(fā)現(xiàn)它的二進(jìn)制O文件沒(méi)有同步更新,而其他C
    發(fā)表于 09-04 10:35

    單片機(jī)hex文件編譯C語(yǔ)言的過(guò)程

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

    hex文件如何查看原c語(yǔ)言代碼

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

    linux驅(qū)動(dòng)程序的編譯方法有哪兩種

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

    可重復(fù)頭文件的固定結(jié)構(gòu)

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

    關(guān)于Makefile自動(dòng)生成-autotools的使用

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

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

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