首頁 >後端開發 >C++ >為什麼我的包含防護無法防止遞歸包含和多個符號定義?

為什麼我的包含防護無法防止遞歸包含和多個符號定義?

Mary-Kate Olsen
Mary-Kate Olsen原創
2024-12-22 19:37:11614瀏覽

Why Are My Include Guards Failing to Prevent Recursive Inclusion and Multiple Symbol Definitions?

為什麼我的包含防護不能防止遞歸包含和多個符號定義?

第一個問題:

為什麼不包含保護我的頭文件免受相互遞歸的影響包容?

包容警衛正在做他們的工作。這個問題是由相互包含的標頭中資料結構定義之間的依賴關係所引起的。

例如,假設 a.h 和 b.h 包含:

// a.h
#ifndef A_H
#define A_H
#include "b.h"
struct A {...};
#endif

// b.h
#ifndef B_H
#define B_H
#include "a.h"
struct B {...};
#endif

如果沒有包含防護,main.cpp 將不會因為編譯深度限製而進行編譯。

使用包含防護,可以防止無限遞歸。然而,定義之間的依賴關係仍然存在:

  • 編譯 a.h 時,預處理器遇到 #include "b.h" 並對其進行處理。
  • 處理 b.h 時,它遇到 #include "a.h" 並處理它。再次處理它。
  • 這會建立一個依賴關係,其中必須先定義A,然後才能定義B

要解決此問題,請在需要它們的標頭中使用前向聲明:

// b.h
#ifndef B_H
#define B_H
struct A; // Forward declaration
struct B {...};
#endif

第二個問題:

為什麼不包含防止多重定義的守衛?

包含守衛可以防止重新定義在單一翻譯單元內,但不能跨不同單元。

例如,當header.h:

#ifndef HEADER_H
#define HEADER_H
int f() {...};
#endif

包含在多個翻譯單元中時,例如source1.cpp 和source2.cpp :

// source1.cpp
#include "header.h"

// source2.cpp
#include "header.h"

連結器會抱怨f()的多個定義,因為它在單獨的物件中看到相同的定義

要解決此問題,請使用以下方法:

  • 內聯關鍵字:指示編譯器內聯函數體,允許在沒有連結器的情況下進行多個定義錯誤。
  • 靜態關鍵字:提供函數內部鏈接,為每個函數建立一個私人副本翻譯單元。
  • 未命名命名空間:在未命名命名空間中聲明函數會為其提供內部鏈接,從而防止多重定義。

以上是為什麼我的包含防護無法防止遞歸包含和多個符號定義?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn