當(dāng)前位置:首頁 > 公眾號精選 > 嵌入式微處理器
[導(dǎo)讀]C語言里,每個源文件是一個模塊,頭文件為使用該模塊的用戶提供接口。接口指一個功能模塊暴露給其他模塊用以訪問具體功能的方法。使用源文件實(shí)現(xiàn)模塊的功能,使用頭文件暴露單元的接口。用戶只需包含相應(yīng)的頭文件就可使用該頭文件中暴露的接口。

說明

本文假定讀者已具備基本的C編譯知識。

如非特殊說明,文中“源文件”指 * .c文件,“頭文件”指 *.h文件,“引用”指包含頭文件。

一、頭文件作用

C語言里,每個源文件是一個模塊,頭文件為使用該模塊的用戶提供接口。接口指一個功能模塊暴露給其他模塊用以訪問具體功能的方法。

使用源文件實(shí)現(xiàn)模塊的功能,使用頭文件暴露單元的接口。用戶只需包含相應(yīng)的頭文件就可使用該頭文件中暴露的接口。

通過頭文件包含的方法將程序中的各功能模塊聯(lián)系起來有利于模塊化程序設(shè)計(jì):

1)通過頭文件調(diào)用庫功能。在很多場合,源代碼不便(或不準(zhǔn))向用戶公布,只要向用戶提供頭文件和二進(jìn)制庫即可。用戶只需按照頭文件中的接口聲明來調(diào)用庫功能,而不必關(guān)心接口如何實(shí)現(xiàn)。編譯器會從庫中提取相應(yīng)的代碼。

2)頭文件能加強(qiáng)類型安全檢查。若某個接口的實(shí)現(xiàn)或使用方式與頭文件中的聲明不一致,編譯器就會指出錯誤。這一簡單的規(guī)則能大大減輕程序員調(diào)試、改錯的負(fù)擔(dān)。

在預(yù)處理階段,編譯器將源文件包含的頭文件內(nèi)容復(fù)制到包含語句(#include)處。在源文件編譯時,連同被包含進(jìn)來的頭文件內(nèi)容一起編譯,生成目標(biāo)文件(.obj)。

如果所包含的頭文件非常龐大,則會嚴(yán)重降低編譯速度(使用GCC的-E選項(xiàng)可獲得并查看最終預(yù)處理完的文件)。因此,在源文件中應(yīng)僅包含必需的頭文件,且盡量不要在頭文件中包含其它頭文件。

二、 頭文件組織原則

源文件中實(shí)現(xiàn)變量、函數(shù)的定義,并指定鏈接范圍。頭文件中書寫外部需要使用的全局變量、函數(shù)聲明及數(shù)據(jù)類型和宏的定義。

建議組織頭文件內(nèi)容時遵循以下原則:

1)頭文件劃分原則:類型定義、宏定義盡量與函數(shù)聲明相分離,分別位于不同的頭文件中。內(nèi)部函數(shù)聲明頭文件與外部函數(shù)聲明頭文件相分離,內(nèi)部類型定義頭文件與外部類型定義頭文件相分離。

注意,類型和宏定義有時無法分拆為不同文件,比如結(jié)構(gòu)體內(nèi)數(shù)組成員的元素個數(shù)用常量宏表示時。因此僅分離類型宏定義與函數(shù)聲明,且分別置于*.th和*.fh文件(并非強(qiáng)制要求)。

2)頭文件的語義層次化原則:頭文件需要有語義層次。不同語義層次的類型定義不要放在一個頭文件中,不同層次的函數(shù)聲明不要放在一個頭文件中。

3)頭文件的語義相關(guān)性原則:同一頭文件中出現(xiàn)的類型定義、函數(shù)聲明應(yīng)該是語義相關(guān)的、有內(nèi)部邏輯關(guān)系的,避免將無關(guān)的定義和聲明放在一個頭文件中。

4)頭文件名應(yīng)盡量與實(shí)現(xiàn)功能的源文件相同,即module.c和module.h。但源文件不一定要包含其同名的頭文件。

5)頭文件中不應(yīng)包含本地?cái)?shù)據(jù),以降低模塊間耦合度。

即只有源文件自己使用的類型、宏定義和變量、函數(shù)聲明,不應(yīng)出現(xiàn)在頭文件里。作用域限于單文件的私有變量和函數(shù)應(yīng)聲明為static,以防止外部調(diào)用。將私有類型置于源文件中,會提高聚合度,并減少不必要的格式外漏。

6)頭文件內(nèi)不允許定義變量和函數(shù),只能有宏、類型(typedef/struct/union/enum等)及變量和函數(shù)的聲明。特殊情況下可extern基本類型的全局變量,源文件通過包含該頭文件訪問全局變量。但頭文件內(nèi)不應(yīng)extern自定義類型(如結(jié)構(gòu)體)的全局變量,否則將迫使本不需要訪問該變量的源文件包含自定義類型所在頭文件[1]。

7)說明性頭文件不需要有對應(yīng)的源文件。此類頭文件內(nèi)大多包含大量概念性宏定義或枚舉類型定義,不包含任何其他類型定義和變量或函數(shù)聲明。此類頭文件也不應(yīng)包含任何其他頭文件。

8)使用#pragma once或header guard(亦稱include guard或macro guard)避免頭文件重復(fù)包含。#pragma once是一種非標(biāo)準(zhǔn)但已被現(xiàn)代編譯器廣泛支持的技巧,它明確告知預(yù)處理器“不要重復(fù)包含當(dāng)前頭文件”。而header guard則通過預(yù)處理命令模擬類似行為:

#ifndef??_PRJ_DIR_FILE_H??//必須確保header?guard宏名永不重名
#define??_PRJ_DIR_FILE_H

//<頭文件內(nèi)容>

#endif


使用#pragma once相比header guard具有兩個優(yōu)點(diǎn):

  • 更快。編譯器不會第二次讀取標(biāo)記#pragma once的文件,但卻會讀若干遍使用header guard 的文件(尋找#endif);

  • 更簡單。不再需要為每個文件的header guard取名,避免宏名重名引發(fā)的“找不到聲明”問題。

    缺點(diǎn)則是:

  • #pragma once保證物理上的同一個文件不會被包含多次,無法對頭文件中的一段代碼作#pragma once聲明。若某個頭文件具有多份拷貝(內(nèi)容相同的多個文件),pragma不能保證它們不被重復(fù)包含。當(dāng)然,這種重復(fù)包含很容易被發(fā)現(xiàn)并修正。

    9) C++中要引用C函數(shù)時,函數(shù)所在頭文件內(nèi)應(yīng)包含extern "C"。

//.h文件頭部
#ifdef??__cplusplus
extern?"C"?{
#endif

//<函數(shù)聲明>

//.h文件尾部
#ifdef??__cplusplus
}
#endif


被extern "C"修飾的變量和函數(shù)將按照C語言方式編譯和連接,否則編譯器將無法找到C函數(shù)定義,從而導(dǎo)致鏈接失敗。

10)頭文件內(nèi)要有面向用戶的充足注釋,從應(yīng)用角度描述接口暴露的內(nèi)容。

三、 頭文件包含原則

在實(shí)際編程中,常常因頭文件包含不當(dāng)而引發(fā)編譯時報告符號未定義的錯誤或重復(fù)定義的警告。要消除符號未定義的編譯錯誤,只需在引用符號(變量、函數(shù)、數(shù)據(jù)類型及宏等)前確保它已被聲明或定義[4]。要消除重復(fù)定義的警告,則需合理設(shè)計(jì)頭文件包含順序和層次。

建議包含頭文件時遵循以下原則:

1)源文件內(nèi)的頭文件包含順序應(yīng)從最特殊到一般,如:

#include?"通用頭文件"??//內(nèi)部可能定義本模塊數(shù)據(jù)類型別名
#include?"源文件同名頭文件"
#include?"本模塊其他頭文件"
#include?"自定義工具頭文件"
#include?"第三方頭文件"
#include?"平臺相關(guān)頭文件"
#include?"C++庫頭文件"
#include?"C庫頭文件"


優(yōu)點(diǎn)是每個頭文件必須include需要的關(guān)聯(lián)頭文件,否則會報錯。同時,源文件同名頭文件置于包含列表前端便于檢查該頭文件是否自完備,以及類型或函數(shù)聲明是否與標(biāo)準(zhǔn)庫沖突。

2)減少頭文件的嵌套和交叉引用,頭文件僅包含其真正需要顯式包含的頭文件。

例如,頭文件A中出現(xiàn)的類型定義在頭文件B中,則頭文件A應(yīng)包含頭文件B,除此以外的其他頭文件不允許包含。

頭文件的嵌套和交叉引用會使程序組織結(jié)構(gòu)和文件組織變得混亂,同時造成潛在的錯誤。大型工程中,原有頭文件可能會被多個其他(源或頭)文件包含,在原有頭文件中添加新的頭文件往往牽一發(fā)而動全身。若頭文件中類型定義需要其他頭文件時,可將其提出來單獨(dú)形成一個全局頭文件。

3)頭文件應(yīng)包含哪些頭文件僅取決于自身,而非包含該頭文件的源文件。

例如,編譯源文件時需要用到頭文件B,且源文件已包含頭文件A,而索性將頭文件B包含在頭文件A中,這是錯誤的做法。

4)盡量保證用戶使用此頭文件時,無需手動包含其他前提頭文件,即此頭文件內(nèi)已包含前提頭文件。

例如,面積相關(guān)操作的頭文件Area.h內(nèi)已包含關(guān)于點(diǎn)操作的頭文件Point.h,則用戶包含Area.h后無需再手動包含Point.h。這樣用戶就不必了解頭文件的內(nèi)在依賴關(guān)系。

5)頭文件應(yīng)是自完備的,即在任一源文件中包含任一頭文件而不會產(chǎn)生編譯錯誤。

6)源文件中包含的頭文件盡量不要有順序依賴。

7)盡量在源文件中包含頭文件,而非在頭文件中。且源文件僅包含所需的頭文件。

8)頭文件中若能前置聲明(亦稱前向聲明[5]),就不要包含另一頭文件。僅當(dāng)前置聲明不能滿足或過于麻煩時才使用include,如此可減少依賴性方面的問題。示例如下:

struct?T_MeInfoMap;??//前置聲明
struct?T_OmciMsg;????//前置聲明

typedef?FUNC_STATUS?(*OmciChkFunc)(struct?T_MeInfoMap?*ptMeInfo,?struct?T_OmciMsg?*ptMsg,?struct?T_OmciMsg?*ptAckMsg);


//OMCI實(shí)體信息
typedef?struct{
????INT16U?wMeClass;???????????????//實(shí)體類別
????OMCI_ATTR_INFO?*pMeAttrInfo;???//實(shí)體所定義的屬性信息指針
????INT8U??ucAttrNum;??????????????//實(shí)體所定義的屬性數(shù)目
????INT16U?wTotalAttrLen;??????????//實(shí)體所有屬性所占的總字節(jié)數(shù),初始化為0,動態(tài)計(jì)算
????INT8U??*pszDbName;?????????????//實(shí)體存庫時的數(shù)據(jù)表名稱,建議不要超過DB_NAME_LEN(32)
????INT16U?wMaxRecNum;?????????????//實(shí)體存庫時支持的最大記錄數(shù)目
????OmciChkFunc?fnCheck;???????????//Omci校驗(yàn)函數(shù)指針
????BOOL???bDbCreated;?????????????//實(shí)體數(shù)據(jù)表是否已創(chuàng)建
}OMCI_ME_INFO_MAP;


如上,在OmciChkFunc函數(shù)的實(shí)現(xiàn)源文件內(nèi)包含T_MeInfoMap和T_OmciMsg所在頭文件即可。

另舉一例如下:

typedef?TBL_SET_MODE?(*OperTypeFunc)(INT8U?*pTblEntry);

typedef?INT8U?(*CmpRecFunc)(VOID?*pvCmpData,?VOID?*pvRecData);?//為避免頭文件交叉引用,與CompareRecFunc異名同構(gòu)

//表屬性信息
typedef?struct{
????INT16U?wMaxEntryNum;?????????//表屬性最大表項(xiàng)數(shù)目(實(shí)體記錄數(shù)目wMaxRecNum?*?wMaxEntryNum?<=?MAX_RECORD_NUM)
????OperTypeFunc?fnGetOperType;??//操作類型函數(shù)指針。根據(jù)表項(xiàng)數(shù)據(jù)或外界需求(只讀表)解析當(dāng)前表項(xiàng)操作類型
????TBL_KEY_INFO?tCmpKeyInfo;????//檢索表屬性子表記錄時的匹配關(guān)鍵字信息(TBL_KEY_INFO)
????CmpRecFunc???fnCmpAddKey;????//增加表項(xiàng)時需要檢測的關(guān)鍵字匹配函數(shù)指針
????CmpRecFunc???fnCmpDelKey;????//刪除表項(xiàng)時需要檢測的關(guān)鍵字匹配函數(shù)指針
????INT16U?wTblEntrySize;????????//表屬性表項(xiàng)字節(jié)數(shù),由外部動態(tài)賦值
}TBL_ATTR_INFO;


如上,CompareRecFunc函數(shù)原型由其他頭文件提供,此處為避免頭文件交叉引用定義其異名同構(gòu)原型CmpRecFunc。

在不會引起歧義的前提下,頭文件內(nèi)盡可能使用VOID指針代替非基本類型的值變量或指針,以避免再包含類型定義所在的頭文件。但這將影響代碼可讀性并降低程序執(zhí)行效率,應(yīng)權(quán)衡利弊。

9)避免包含重量級的平臺頭文件,如windows.h或d3d9.h等。若僅使用該頭文件少量函數(shù),可extern函數(shù)到源文件內(nèi)。如下:

/****************************************************************************************
???????????????????????外部函數(shù)聲明?(當(dāng)外部接口未提供頭文件或頭文件過于復(fù)雜時)?
?****************************************************************************************/

//因聲明所在頭文件引用混亂,此處僅extern函數(shù)聲明。
extern?INT32S?DBShmCliInit(VOID);?//#include?"db_shm_mgr.h"
extern?INT32S?cmLockInit(VOID);???//#include?"common_cmapi.h"


若還使用該頭文件某些類型和宏定義,可創(chuàng)建適配性源文件。在該源文件內(nèi)包含平臺頭文件,封裝新的接口并將其聲明在同名頭文件內(nèi),其他源文件將通過適配頭文件間接訪問平臺接口。如下:

/*****************************************************************************************
*?文件名稱:Omci_Send_Msg.c
*?內(nèi)容摘要:OMCI消息轉(zhuǎn)發(fā)接口
*?其它說明:?該頭文件封裝SEND接口,以避免其他源文件包含支撐api和pid公共頭文件導(dǎo)致引用混亂。
?*****************************************************************************************/


#include?"Omci_Common.h"
#include?"Omci_Send_Msg.h"
#include?"oss_api.h"

/**********************************************************************************************
?????????????????????????????????????????函數(shù)實(shí)現(xiàn)區(qū)
**********************************************************************************************/


//向自身進(jìn)程發(fā)送異步消息
INT32U?OmciAsynSendSelf(INT16U?wEvent,?VOID?*pvMsg,?INT16U?wMsgLen)
{
????PID?dwSelfPid?=?0;
????SELF(&dwSelfPid);
????return?ASEND(wEvent,?pvMsg,?wMsgLen,?dwSelfPid);
}


10)對于函數(shù)庫(包括標(biāo)準(zhǔn)庫和自定義的公共宏及接口)的頭文件,可將其加入到一個通用頭文件中。需要控制該頭文件的體積(主要是該頭文件所包含的所有頭文件內(nèi)容大小),并確保所有源文件首先包含該通用頭文件。示例如下:

#ifndef??_OMCI_COMMON_H
#define??_OMCI_COMMON_H

/*******************************************************************************************
*?說明:
*?本文件僅應(yīng)包含與具體通信協(xié)議無關(guān)的通用數(shù)據(jù)類型及宏定義。
*?為簡化頭文件包含且不失可移植性,本文件內(nèi)可包含少量C庫通用頭文件。
*?因本文件內(nèi)定義基本數(shù)據(jù)類型別名,故.c文件中應(yīng)將本頭文件置于包含列表頂端,
*?否則編譯時可能產(chǎn)生類型未定義錯誤。
*******************************************************************************************/


#include?
#include?
#include?
#include?
#include?

#include?"Omci_Byte.h"

//


注意,示例頭文件內(nèi)包含C庫文件雖能簡化包含,但卻與規(guī)則1沖突。也可另外增加包含庫文件列表的通用頭文件。

11)若不確定類型、宏定義或函數(shù)聲明所在頭文件具體路徑,可在源文件中再次定義或聲明,編譯器會以redefined警告或conflicting錯誤給出類型、宏定義或函數(shù)聲明所在頭文件路徑。

四、代碼文件組織原則

建議C語言項(xiàng)目中代碼文件組織遵循以下原則:

1)使用層次化和模塊化的軟件開發(fā)模型。每個模塊只能使用所在層和下一層模塊提供的接口。

2)每個模塊的文件(可能多個)保存在一個獨(dú)立文件夾中。

模塊文件較多時可采用子目錄的方式,物理上隔離不同層次的文件。子目錄下源文件和頭文件應(yīng)分開存放,如分別置入include和source目錄。

3)用于模塊裁減的條件編譯宏保存在一個獨(dú)立文件中,便于軟件裁減。

4)硬件相關(guān)代碼和操作系統(tǒng)相關(guān)代碼與工程代碼相對獨(dú)立保存,以便于軟件移植。

5)按相同功能或相關(guān)性組織源文件和頭文件。同一文件內(nèi)的聚合度要高,不同文件中的耦合度要低。

在對既有工程做單元測試時,耦合度低的文件布局非常便于搭建環(huán)境。

6)聲明和定義分開,使用頭文件暴露模塊需要提供給外部的類型、宏、變量和函數(shù)。盡量做到模塊對外部透明,用戶在使用模塊功能時無需了解具體的實(shí)現(xiàn)。

7)作為對外接口的頭文件一經(jīng)發(fā)布,應(yīng)保持穩(wěn)定。修改時一定要慎重。

8)文件夾和文件命名要能夠反映出模塊的功能。

9)正式版本和測試版本使用統(tǒng)一文件,使用宏控制是否產(chǎn)生測試輸出。

10)必要的注釋不可缺少。

五、 注解

「【注1】全局變量的使用原則」

1)若全局變量僅在單個源文件中訪問,則可將該變量改為該文件內(nèi)的靜態(tài)全局變量;

2)若全局變量僅由單個函數(shù)訪問,則可將該變量改為該函數(shù)內(nèi)的靜態(tài)局部變量;

3)盡量不要使用extern聲明全局變量,最好提供函數(shù)訪問這些變量。直接暴露全局變量是不安全的,外部用戶未必完全理解這些變量的含義。

4)設(shè)計(jì)和調(diào)用訪問動態(tài)全局變量、靜態(tài)全局變量、靜態(tài)局部變量的函數(shù)時,需要考慮重入問題。

「【注2】#pragma once的可移植性」

#ifndef由C/C++語言標(biāo)準(zhǔn)支持,不受編譯器任何限制;而#pragma once僅由編譯器提供保證,存在可移植性等問題。

某些gcc編譯器版本(如3.2.3)會報告“warning: #pragma once is obsolete”的警告,而其他較老版本的編譯器可能會報錯。但隨著gcc 3.4的發(fā)布,#pragma once中的一些問題(主要與符號鏈接和硬鏈接有關(guān))得以解決,#pragma once命令也標(biāo)記為“未廢棄”。

還有種寫法同時使用#pragma once和header guard編寫“可移植性”代碼,以利用編譯器可能支持的#pragma once優(yōu)化。如下:

#pragma?once
#ifndef?_PRJ_DIR_FILE_H
#define?_PRJ_DIR_FILE_H

//<頭文件內(nèi)容>

#endif


該法似乎兼有兩者的優(yōu)點(diǎn)。但既然使用#ifndef就有宏名重名的風(fēng)險,也無法避免不支持#pragma once的編譯器告警或報錯,故混用兩種方法似乎不能帶來更多的好處,反倒讓不熟悉的人感到困惑。

注意,如果使用header guard,理論上可在代碼任何地方判斷當(dāng)前是否已經(jīng)包含某個頭文件。但應(yīng)避免通過該判斷來改變后續(xù)代碼的邏輯走向!

這種做法將使程序依賴于頭文件的包含順序,極不可取。若需要實(shí)現(xiàn)“若當(dāng)前包含HeaderA.h,才加入StructB結(jié)構(gòu)”,可對StructB結(jié)構(gòu)創(chuàng)建HeaderB.h頭文件,在HeaderA.h中包含HeaderB.h。

「【注3】extern "C"」

C++語言在編譯時為實(shí)現(xiàn)函數(shù)重載,會結(jié)合函數(shù)名、參數(shù)數(shù)目及類型信息而生成一個中間函數(shù)名。

例如,C++中函數(shù)void foo(int x, float y)編譯后在符號庫中生成的名字為_foo_int_float(不同編譯器可能生成不同函數(shù)名,但均采用相同機(jī)制,生成的新名字稱為”mangled name”);而該函數(shù)被C編譯器編譯后在符號庫中的名字為_foo。

C語言中不支持extern "C"聲明,在.c文件中包含extern "C"時會出現(xiàn)編譯語法錯誤。

當(dāng)然編譯器也可以為其他語言提供鏈接說明。例如:extern "FORTRAN"、extern "Ada"等。

「【注4】聲明(declaration)與定義(definition)」

全局變量或函數(shù)可(在多個編譯單元中)有多處聲明,但只允許定義一次。全局變量定義時分配空間并賦初始值(如果有);函數(shù)定義時提供函數(shù)體內(nèi)容。

聲明:
extern?int?iGlobal;
extern?int?func();?或int?func();

定義:
int?iGlobal?=?0;?或int?iGlobal;
int?func?()
{
????return?1;
}


在多個源文件中共享變量或函數(shù)時,需確保定義和聲明的一致性。通常在某個相關(guān)的源文件中定義,然后在頭文件中進(jìn)行外部聲明。需要使用時包含相應(yīng)的頭文件即可。定義變量的源文件也應(yīng)包含該頭文件,以便編譯器檢查定義和聲明的一致性。

該規(guī)則可提供高度的可移植性:它與ANSI/ISO C標(biāo)準(zhǔn)一致,同時也兼顧大多數(shù)ANSI前的編譯器和鏈接器。(Unix編譯器和鏈接器常使用允許多重定義的“通用模式”,只要保證最多對一處定義進(jìn)行初始化即可。

該方式被ANSI C標(biāo)準(zhǔn)稱為一種“通用擴(kuò)展”)。某些很老的系統(tǒng)可能要求顯式初始化以區(qū)別定義和外部聲明。

通用擴(kuò)展在《深入理解計(jì)算機(jī)系統(tǒng)》中解釋為:多重定義的符號只允許最多一個強(qiáng)符號。函數(shù)和定義時已初始化的全局變量是強(qiáng)符號;未初始化的全局變量是弱符號。Unix鏈接器使用以下規(guī)則來處理多重定義的符號:

規(guī)則一:不允許有多個強(qiáng)符號。在被多個源文件包含的頭文件內(nèi)定義的全局變量會被定義多次(預(yù)處理階段會將頭文件內(nèi)容展開在源文件中),若在定義時顯式地賦值(初始化),則會違反此規(guī)則。

規(guī)則二:若存在一個強(qiáng)符號和多個弱符號,則選擇強(qiáng)符號。

規(guī)則三:若存在多個弱符號,則從這些弱符號中任選一個。

當(dāng)不同文件內(nèi)定義同名(即便類型和含義不同)的全局變量時,該變量共享同一塊內(nèi)存(地址相同)。若變量定義時均初始化,則會產(chǎn)生重定義(multiple definition)的鏈接錯誤;若某處變量定義時未初始化,則無鏈接錯誤,僅在因類型不同而大小不同時可能產(chǎn)生符號大小變化(size of symbol `XXX' changed)的編譯警告。

在最壞情況下,編譯鏈接正常,但不同文件對同名全局變量讀寫時相互影響,引發(fā)非常詭異的問題。這種風(fēng)險在使用無法接觸源碼的第三方庫時尤為突出。

因此,應(yīng)盡量避免使用全局變量。若確有必要,應(yīng)采用靜態(tài)全局變量(無強(qiáng)弱之分,且不會和其他全局符號產(chǎn)生沖突),并封裝訪問函數(shù)供外部文件調(diào)用。

「【注5】前向聲明(forward declaration)」

結(jié)構(gòu)體類型S在聲明之后定義之前是一個不完全類型(incomplete type),即已知S是一個類型,但不知道包含哪些成員。

不完全類型只能用于定義指向該類型的指針,或聲明使用該類型作為形參指針類型或返回指針類型的函數(shù)。指針類型對編譯器而言大小固定(如32位機(jī)上為四字節(jié)),不會出現(xiàn)編譯錯誤。

假設(shè)先后定義兩個結(jié)構(gòu)A和B,且兩個結(jié)構(gòu)需要互相引用。在定義A時B還沒有定義,則要引用B就需要前向聲明結(jié)構(gòu)B(struct B;)。示例如下:

typedef?BOOL?(*func)(const?DefStruct?*ptStrt);
?
typedef?struct?DefStruct_t
{

????int?i;
????func?f;
}DefStruct;


如上在DefStruct中使用回調(diào)函數(shù)func聲明,這樣交叉引用必然編譯報錯。進(jìn)行前向聲明即可:

typedef?struct?DefStruct_t?DefStruct;
typedef?BOOL?(*func)(const?DefStruct?*ptStrt);

struct?DefStruct_t
{

????int?i;
????func?f;
};

注意,在前向聲明和具體定義之間涉及標(biāo)識符(變量、結(jié)構(gòu)、函數(shù)等)實(shí)現(xiàn)細(xì)節(jié)的使用都是非法的。若函數(shù)被前向聲明但未被調(diào)用,則編譯和運(yùn)行正常;若前向聲明函數(shù)被調(diào)用但未被定義,則編譯正常但鏈接報錯(undefined reference)。將具體定義放在源文件中可部分避免該問題。

-END-

原文 | https://www.cnblogs.com/clover-toeic/p/3728026.html

作者 |?clover_toeic


|?整理文章為傳播相關(guān)技術(shù),版權(quán)歸原作者所有?|

|?如有侵權(quán),請聯(lián)系刪除?|


【1】超長干貨為你解析:從串口驅(qū)動到Linux驅(qū)動模型,嵌入式必會!

【2】超全!嵌入式必懂的CAN總線一文講通了

【3】干貨:嵌入式系統(tǒng)設(shè)計(jì)開發(fā)大全?。ㄈf字總結(jié))

【4】嵌入式系統(tǒng)求職回憶錄:廣嵌、邁瑞、華為、智光……

【5】嵌入式行業(yè)真的沒有前途嗎?



免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點(diǎn),不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

嵌入式ARM

掃描二維碼,關(guān)注更多精彩內(nèi)容

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉