自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

頭文件循環(huán)引用:如何破解這個(gè)編程死循環(huán)?

開發(fā)
在這趟奇妙的C++頭文件之旅中,我們一起探討了如何優(yōu)雅地引入頭文件,就像在圖書館借書一樣簡單有趣。

嘿,讓我們來聊聊C++中那些可愛的頭文件引入方式吧! 

當(dāng)我們在代碼中看到#include 時(shí),你是否注意到它后面可以跟著兩種不同的"穿搭" —— 尖括號(hào)<> 和雙引號(hào)""??? 這可不是隨便選的哦!

想象一下,尖括號(hào)<> 就像是去圖書館借書 ??,系統(tǒng)會(huì)先去"公共書架"(編譯器的標(biāo)準(zhǔn)路徑)找,找不到再去"特藏室"(系統(tǒng)變量路徑)翻找。這種方式通常用來引入那些標(biāo)準(zhǔn)庫文件,比如我們常見的<iostream> 和<string> 。

而雙引號(hào)"" 則更像是在自己家里找書 ??,它會(huì)先在"書房"(當(dāng)前文件目錄)翻找,找不到才會(huì)去"圖書館"(編譯器路徑和系統(tǒng)變量)借閱。這種方式主要用于我們自己編寫的頭文件,就像是我們自己的私人筆記本一樣。

頭文件引入小故事

來來來,讓我給你講個(gè)有趣的故事!想象一下,在C++的世界里,引入頭文件就像是在圖書館借書一樣有趣 ??

當(dāng)我們需要標(biāo)準(zhǔn)庫的時(shí)候,就像去大圖書館的公共區(qū)域借書一樣 ???,我們會(huì)這樣寫:

#include <iostream>     // 借本輸入輸出的魔法書 ?
#include <string>       // 再來本字符串變變變 ??
#include <vector>       // 這本是動(dòng)態(tài)數(shù)組的秘籍 ??
#include <algorithm>    // 最后來本算法寶典 ??

但是呢,有時(shí)候我們也需要用自己寫的"私房菜譜"(自定義頭文件)??,這時(shí)候就要用雙引號(hào)來"翻看"啦:

#include "MyClass.h"           // 就在書桌上的筆記本 ??
#include "utils/helpers.h"     // 放在工具箱里的說明書 ???
#include "../common/config.h"  // 樓上收藏的配置手冊 ??

看,是不是感覺頭文件引入也可以這么有趣呢??? 記住啦,標(biāo)準(zhǔn)庫就像公共圖書館的藏書,用尖括號(hào)<> 來借閱;而自己的小筆記就用雙引號(hào)"" 來翻看,就像在自己的書房里找書一樣方便!??

頭文件查找過程詳解

讓我們以#include <iostream> 為例,一起來探索編譯器是如何查找和引入頭文件的奇妙過程吧! ??

1. 使用尖括號(hào)<> 的查找過程 ??

當(dāng)我們寫下#include <iostream> 時(shí),編譯器會(huì)像偵探一樣按以下順序仔細(xì)查找 ???♂?:

(1) 標(biāo)準(zhǔn)庫目 ?? 編譯器在安裝時(shí)就預(yù)先配置了標(biāo)準(zhǔn)庫的搜索路徑。這些神奇的路徑是如何確定的呢?

a.編譯器安裝時(shí)的配置 ??

# GCC編譯器 ???
g++ -v -E -x c++ /dev/null

# Clang編譯器 ??
clang++ -v -E -x c++ /dev/null

b.默認(rèn)搜索路徑 ???

# 在 Linux/Unix 系統(tǒng)中通常是: ??
/usr/include/c++/<版本號(hào)>      # C++標(biāo)準(zhǔn)庫頭文件 ??
/usr/local/include            # 本地安裝的庫文件 ??
/usr/include                  # 系統(tǒng)級別的頭文件 ??

# 在 Windows + MSVC 中通常是: ??
C:\Program Files (x86)\Microsoft Visual Studio\<版本>\<版本號(hào)>\VC\include

c.搜索順序的原理 

例如,當(dāng)你包含<iostream> 時(shí)的實(shí)際過程:

#include <iostream>

// 1. 編譯器首先在內(nèi)置緩存中找iostream
// 2. 如果沒找到,則在/usr/include/c++/<版本號(hào)>/iostream查找
// 3. 找到后,檢查是否已經(jīng)被包含(通過頭文件保護(hù)符)
// 4. 如果是首次包含,則讀取并處理文件內(nèi)容

當(dāng)我們安裝C++編譯器時(shí),安裝程序會(huì)自動(dòng)設(shè)置標(biāo)準(zhǔn)庫的位置,這些位置被硬編碼到編譯器的配置文件中。

可以通過以下魔法咒語查看編譯器的搜索路徑:

  • 編譯器首先檢查內(nèi)置的頭文件緩存(如果有的話) ??
  • 然后按照預(yù)定義的搜索路徑順序查找 ??
  • 最后查找環(huán)境變量指定的路徑 ??

(2) 為什么要使用這種搜索機(jī)制?

  • 安全性:系統(tǒng)頭文件存放在受保護(hù)的目錄中,防止意外修改
  • 統(tǒng)一性:所有項(xiàng)目都使用相同版本的標(biāo)準(zhǔn)庫,確保兼容性
  • 效率:預(yù)定義的搜索路徑可以加快文件查找速度
  • 維護(hù)性:系統(tǒng)升級時(shí)只需更新中央位置的文件

(3) 如何查看具體的頭文件內(nèi)容?

# 在Linux系統(tǒng)中可以直接查看
cat /usr/include/c++/<版本號(hào)>/iostream

# 或者使用編譯器顯示預(yù)處理后的內(nèi)容
g++ -E myfile.cpp | less

2. 使用雙引號(hào)"" 的查找過程 ??

以#include "myproject.h" 為例 ??:

  • 首先在當(dāng)前源文件所在目錄查找 ??
// 如果源文件在 src/main.cpp
#include "myproject.h"  // 會(huì)先查找 src/myproject.h ??
  • 然后查找相對路徑 ???
// 在 src/main.cpp 中
#include "../include/myproject.h"  // 查找上級目錄的 include 文件夾 ??
  • 最后按照尖括號(hào)<> 的查找規(guī)則繼續(xù)查找 ??

頭文件引入的實(shí)際過程

讓我們看一個(gè)完整的例子 ??:

// main.cpp ??
#include <iostream>
#include "utils/math_helper.h"

int main() {
    // ...
}

預(yù)處理器處理這個(gè)文件的步驟 ??:

  • 展開<iostream> ??
// 1. 在標(biāo)準(zhǔn)庫路徑找到 iostream ??
// 2. 檢查是否已經(jīng)包含(通過頭文件保護(hù)符)???
// 3. 展開內(nèi)容,例如:?
namespace std {
    class ios_base { /*...*/ };  // 基礎(chǔ)輸入輸出類 ??
    class istream { /*...*/ };   // 輸入流類 ??
    // ...
}
  • 展開"utils/math_helper.h" ??
// 1. 先在當(dāng)前目錄查找 utils/math_helper.h ??
// 2. 如果找不到,繼續(xù)在編譯器指定的路徑查找 ???
// 3. 展開內(nèi)容 ??

項(xiàng)目實(shí)踐中的頭文件組織

在實(shí)際項(xiàng)目中,推薦這樣組織頭文件 ???:

project/ ?? ├── include/ # 公共頭文件目錄 ?? │ ├── project/ # 項(xiàng)目頭文件 ?? │ │ ├── core.h # 核心頭文件 ? │ │ └── utils.h # 工具頭文件 ??? │ └── third_party/ # 第三方庫頭文件 ?? ├── src/ # 源文件目錄 ?? │ ├── core.cpp # 核心實(shí)現(xiàn) ?? │ └── utils.cpp # 工具實(shí)現(xiàn) ?? └── CMakeLists.txt # CMake 構(gòu)建文件 ???

使用時(shí) ????:

// 在 src/core.cpp 中 ??
#include "project/core.h"    // 使用項(xiàng)目頭文件 ??
#include <algorithm>         // 使用標(biāo)準(zhǔn)庫 ??

所以下次當(dāng)你在寫代碼時(shí),記住這個(gè)簡單的規(guī)則 ??:

  • 系統(tǒng)的標(biāo)準(zhǔn)庫文件就用尖括號(hào)<> ??#include <iostream> ??
  • 自己寫的頭文件就用雙引號(hào)"" ??#include "myheader.h" ??

就是這么簡單又合理! ? 讓我們的代碼結(jié)構(gòu)更清晰、更優(yōu)雅! ??

頭文件循環(huán)引用:一個(gè)有趣的解決方案

嘿,小伙伴們!?? 今天讓我們來聊一個(gè)在 C++ 開發(fā)中經(jīng)常遇到的"死循環(huán)"難題 ??

想象一下,就像兩個(gè)好朋友互相依賴的情況 ?? —— PersonA 想認(rèn)識(shí) PersonB,而 PersonB 也想認(rèn)識(shí) PersonA。在代碼世界里,這種情況可能會(huì)讓編譯器陷入混亂 ??

來看看這個(gè)有趣的例子:

// 文件:header1.h
#include "header2.h"
class PersonA {
private:
    PersonB* m_friend;  // 想和 PersonB 做朋友 ??
public:
    void sayHello();
};

// 文件:header2.h
#include "header1.h"
class PersonB {
private:
    PersonA* m_friend;  // 也想和 PersonA 做朋友 ??
public:
    void greet();
};

哎呀!這樣寫代碼就像兩個(gè)人互相追著對方的尾巴轉(zhuǎn)圈圈 ??,編譯器看到這種情況就會(huì)抓狂: "咦?要先編譯誰呢?" ??

不過別擔(dān)心!我們有一個(gè)聰明的解決方案 ? —— 就是使用"前向聲明"這個(gè)魔法咒語 ?? 告訴編譯器:"嘿,相信我,這個(gè)類待會(huì)兒就來!"

就像這樣改寫:

// 文件:header1.h
#ifndef HEADER1_H
#define HEADER1_H

class PersonB;  // 先說好:PersonB 待會(huì)兒就來!?
class PersonA {
    // ... 其他代碼保持不變 ...
};

#endif

這樣一來,我們的代碼就像一場優(yōu)雅的舞會(huì) ????,每個(gè)類都能找到自己的舞伴,編譯器也不會(huì)暈頭轉(zhuǎn)向啦!記住,有時(shí)候編程就像交朋友,不要太著急,慢慢來,總會(huì)遇到對的那個(gè)人(啊不,是類 ??)!

那如果不是指針引用呢?

有時(shí)候,我們可能會(huì)遇到需要直接引用對象而不是指針的情況:

// 文件:header1.h
#include "header2.h"
class PersonA {
private:
    PersonB m_friend;  // 想直接把朋友裝進(jìn)口袋!??
public:
    void sayHello();
};

// 文件:header2.h
#include "header1.h"
class PersonB {
private:
    PersonA m_friend;  // 我也要把朋友裝進(jìn)口袋!
public:
    void greet();
};

哎呀!這下可有意思了!?? 編譯器看到這段代碼時(shí)就像是在解一個(gè)"先有雞還是先有蛋"的問題 ????

為什么呢?讓我們來演一出小品:想象編譯器是一位可愛的搬家工人 ??

搬家工人:「嗯,讓我看看要搬的東西...PersonA類需要多大的空間呢?」 ?? 「哦,它里面有個(gè)PersonB,得先知道PersonB多大」 ?? 「那讓我看看PersonB...咦?它里面又有個(gè)PersonA?」 ?? 「但我還不知道PersonA多大啊...」 ?? 「但要知道PersonA多大,我得先知道PersonB多大...」 ?? 就這樣無限循環(huán)下去啦!

這就像是兩個(gè)小朋友互相說:"我要做一個(gè)和你一樣大的餅干!" "不,我要做一個(gè)和你的餅干一樣大的餅干!" ?? 最后誰也不知道該做多大的餅干才對!??

這就是為什么前向聲明在這種情況下幫不上忙 - 因?yàn)榫幾g器需要知道類的具體大小才能分配內(nèi)存。用指針的話就不同啦,指針就像是一張藏寶圖 ???,大小是固定的,不管藏寶箱(對象)有多大!

所以下次如果你遇到這種情況,記得要么用指針(藏寶圖)???,要么用智能指針(帶GPS定位的藏寶圖)??,要么就得重新???計(jì)你的類結(jié)構(gòu)咯!就像重新安排兩個(gè)小朋友的玩具收藏方式一樣!??

接口分離

不過別擔(dān)心,我們有個(gè)超棒的解決方案 - 接口分離!它就像是給小朋友們發(fā)了一張"交友名片"一樣 ??。這張名片上只寫著最重要的信息:"我會(huì)打招呼!",而不用把所有細(xì)節(jié)都告訴對方。

來看個(gè)具體的例子:

// 先設(shè)計(jì)一張可愛的交友名片 ??
class IFriend {
    virtual void sayHi() = 0;     // 我會(huì)說"嗨!" ??
    virtual void share() = 0;     // 我會(huì)分享玩具! ??
    virtual ~IFriend() = default; // 記得要好好說再見 ??
};

// 小明拿著這張名片來交朋友 ??
class XiaoMing : public IFriend {
    void sayHi() override { 
        std::cout << "嗨,我是小明!" << std::endl; 
    }
    void share() override {
        std::cout << "給你我的變形金剛!" << std::endl;
    }
};

// 小紅也想交朋友 ??
class XiaoHong {
private:
    IFriend& myFriend;  // 只需要知道對方有張交友名片就夠啦!
public:
    void playWith() {
        myFriend.sayHi();    // 和朋友打招呼 ??
        myFriend.share();    // 一起分享玩具 ??
    }
};

看!通過這種方式,小明和小紅就能愉快地玩耍了,完全不用擔(dān)心"我需要先認(rèn)識(shí)你,還是你需要先認(rèn)識(shí)我"這樣的煩惱 ??。這就是接口分離的魔力 ? - 它讓我們的代碼世界變得更簡單,更有趣!

記住啦,當(dāng)你遇到循環(huán)引用的困擾時(shí),就想想這個(gè)可愛的交友名片故事吧!讓代碼像小朋友們一樣,輕松快樂地交朋友!?? 這就是接口分離的精髓所在!??

總結(jié)

嘿,親愛的代碼冒險(xiǎn)家們!?? 在這趟奇妙的C++頭文件之旅中,我們一起探討了如何優(yōu)雅地引入頭文件,就像在圖書館借書一樣簡單有趣 ??。記住,標(biāo)準(zhǔn)庫文件就像公共圖書館的藏書,用尖括號(hào)<>來借閱,而自己的小筆記就用雙引號(hào)""來翻看,就像在自己的書房里找書一樣方便!??

當(dāng)然啦,頭文件的循環(huán)引用就像兩個(gè)小朋友互相追著對方的尾巴轉(zhuǎn)圈圈 ??,但別擔(dān)心,我們有聰明的解決方案,比如用前向聲明這個(gè)魔法咒語 ??,或者用接口分離的交友名片 ??,讓代碼世界變得更簡單,更有趣!?

責(zé)任編輯:趙寧寧 來源: everystep
相關(guān)推薦

2024-12-04 09:47:26

C++頭文件實(shí)現(xiàn)類

2013-06-06 13:34:56

HashMap線程不安全

2011-09-07 10:13:04

IPv6IPv4

2018-10-10 20:20:14

2020-12-17 07:39:30

HashMap死循環(huán)數(shù)據(jù)

2025-01-21 00:00:00

HashMap死循環(huán)數(shù)據(jù)損壞

2022-01-20 08:44:25

HashMap死循環(huán)開放性

2011-08-29 16:23:29

Lua腳本

2022-01-18 06:59:50

HashMap循環(huán)底層

2010-04-26 13:30:21

服務(wù)器虛擬化

2020-09-29 15:24:07

面試數(shù)據(jù)結(jié)構(gòu)Hashmap

2022-06-18 23:10:56

前端模塊循環(huán)依賴

2010-03-11 14:15:24

Python循環(huán)

2020-05-27 12:45:52

HashMapJava加載因子

2019-12-26 12:47:10

BashLinux命令

2009-07-24 17:43:35

循環(huán)引用ASP.NET AJA

2021-10-27 07:15:36

Go 循環(huán)引用

2017-05-04 20:15:51

iOSNSTimer循環(huán)引用

2018-06-29 09:06:18

創(chuàng)業(yè)公司事業(yè)

2023-01-31 08:24:55

HashMap死循環(huán)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)