1. 程式人生 > >為什麼要使用預編譯頭

為什麼要使用預編譯頭

許多初學 VC 的朋友也許都為那麼一個問題困擾過:

    為什麼所有的 cpp 都必須 #include “stdafx.h”

    也許請教了別的高手之後,他們會告訴你,這是預編譯頭,必須包含。可是,這到底
是為什麼呢?預編譯頭有什麼用呢?

    這得從標頭檔案的編譯原理講起。其實標頭檔案並不神祕,它的全部作用,就是把自己的
所有內容直接“貼上”到相應的 #include 語句處。如果不相信的話,不妨做個實驗,將
一個 cpp 中的所有 #include 語句刪掉,並將它包含的檔案貼上到相應的位置,你會發
現,檔案的編譯和執行都完全沒有受到影響。其實,編譯器在編譯你的程式的時候,所做
的第一件事,也就是展開所有的 #include 語句和 #define 語句。

    標頭檔案的出現,固然給書寫程式帶來了很大方便。可是到了 Windows 時代後,慢慢
就呈現出一些問題了。幾乎所有的 Windows 程式都必須包含 windows.h,而那個檔案卻
碩大無比,將它展開後往所有檔案中一貼上,編譯的時候立刻慢得像只蝸牛。

    到了 MFC 時代後,情況更為惡劣了。畢竟 C 風格的 Windows 標頭檔案裡面包含的還
僅僅是函式定義和巨集,編譯難度不算太大,而 MFC 庫裡面的標頭檔案可都是類宣告啊!更
何況,一個最簡單的工程,都會生成大量的類,需要用到大量的函式。如果工程稍微複雜
一些,編譯難度可想而知!

    但是,人們驚奇地發現,雖然用到的標頭檔案又多又雜,但是在一個工程中,總有那麼
一堆標頭檔案,是幾乎所有 cpp 都必須包含的。那麼,可不可以把這些標頭檔案提取出來,
只編譯一編,然後所有其它 cpp 就都能使用呢?沒錯,這就是預編譯頭的思想都由來!

    實踐證明,使用了預編譯頭技術後,編譯速度大大提高了。可以到你的工程目錄下的
Debug 或 Release 目錄中看一看,裡面有一個體積極為碩大的 .pch 檔案,那就是傳說
中的“編譯之後的預編譯頭”。

    使用了預編譯頭技術後,雖然帶來了極大地方便,但也造成了一個問題:由於它假定
預編譯頭中包含過的標頭檔案會在所有 cpp 中使用,因此它在編譯你的 cpp 的時候,就會
將預編譯頭中已經編譯完的部分載入到記憶體中。如果它突然發現你的 cpp 居然沒有包含
預編譯頭,它就會很鬱悶,因為它不知道該如何將已編譯完的部分從記憶體中請出去,整個
編譯過程就會失敗。

    因此,如果你使用了預編譯頭技術,就必須在所有的 cpp 中包含預編譯頭。MFC 工
程中為你建立了一個預設的預編譯頭 stdafx.h,如果你願意,也可以在自己的工程中使
用其它檔名作為你的預編譯頭,如果你覺得有必要。
 

預編譯標頭檔案的使用 
關鍵字:預編譯,/Yu,/Yc,/Yx
本文介紹VC6的預編譯功能的使用,由於預編譯詳細使用比較的複雜,這裡只介紹幾個最重要的預編譯指令: /Yu, /Yc,/Yx,/Fp。其它的詳細資料可以參考:
      MSDN->Visual Studio D6.0Document -> Visual C++6.0 Document
         ->VC++ Programmer Guider ->Compiler and Linker
         ->Details->Creating Precompiled Header files

預編譯頭的概念:
所謂的預編譯頭就是把一個工程中的那一部分程式碼,預先編譯好放在一個檔案裡(通常是以.pch為副檔名的),這個檔案就稱為預編譯標頭檔案這些預先編譯好的程式碼可以是任何的C/C++程式碼——–甚至是inline的函式,但是必須是穩定的,在工程開發的過程中不會被經常改變。如果這些程式碼被修改,則需要重新編譯生成預編譯標頭檔案。注意生成預編譯標頭檔案是很耗時間的。同時你得注意預編譯標頭檔案通常很大,通常有6-7M大。注意及時清理那些沒有用的預編譯標頭檔案。
也許你會問:現在的編譯器都有Time stamp的功能,編譯器在編譯整個工程的時候,它只會編譯那些經過修改的檔案,而不會去編譯那些從上次編譯過,到現在沒有被修改過的檔案。那麼為什麼還要預編譯標頭檔案呢?答案在這裡,我們知道編譯器是以檔案為單位編譯的,一個檔案經過修改後,會重新編譯整個檔案,當然在這個檔案裡包含的所有標頭檔案中的東西(.eg Macro, Preprocessor )都要重新處理一遍。VC的預編譯標頭檔案儲存的正是這部分資訊。以避免每次都要重新處理這些標頭檔案。
預編譯頭的作用:
方法一:手動方法
根據上文介紹,預編譯標頭檔案的作用當然就是提高便宜速度了,有了它你沒有必要每次都編譯那些不需要經常改變的程式碼。編譯效能當然就提高了。
預編譯頭的使用:
     要使用預編譯頭,我們必須指定一個頭檔案,這個標頭檔案包含我們不會經常改變的程式碼和其他的標頭檔案,然後我們用這個標頭檔案來生成一個預編譯標頭檔案(.pch檔案)
 想必大家都知道 StdAfx.h這個檔案。很多人都認為這是VC提供的一個“系統級別”的,編譯器帶的一個頭檔案。其實不是的,這個檔案可以是任何名字的。我們來考察一個典型的由AppWizard生成的MFC Dialog Based 程式的預編譯標頭檔案。(因為AppWizard會為我們指定好如何使用預編譯標頭檔案,預設的是StdAfx.h,這是VC起的名字)。我們會發現這個標頭檔案裡包含了以下的標頭檔案:
#include <afxwin.h>         // MFC core and standard components
#include <afxext.h>         // MFC extensions
#include <afxdisp.h>        // MFC Automation classes
#include <afxdtctl.h>             // MFC support for Internet Explorer 4 Common Controls
#include <afxcmn.h>     
這些正是使用MFC的必須包含的標頭檔案,當然我們不太可能在我們的工程中修改這些標頭檔案的,所以說他們是穩定的。
那麼我們如何指定它來生成預編譯標頭檔案。我們知道一個頭檔案是不能編譯的。所以我們還需要一個cpp檔案來生成.pch 檔案。這個檔案預設的就是StdAfx.cpp。在這個檔案裡只有一句程式碼就是:#include “Stdafx.h”。原因是理所當然的,我們僅僅是要它能夠編譯而已―――也就是說,要的只是它的.cpp的副檔名。我們可以用/Yc編譯開關來指定StdAfx.cpp來生成一個.pch檔案,通過/Fp編譯開關來指定生成的pch檔案的名字。開啟project ->Setting->C/C++ 對話方塊。把Category指向Precompiled Header。在左邊的樹形視圖裡選擇整個工程 (如圖)
(圖1)
在圖中我們的Project Options(右下角的那個白的地方)可以看到 /Fp “debug/PCH.pch”,這就是指定生成的.pch檔案的名字,預設的通常是 <工程名>.pch(我的示例工程名就是PCH)。
然後,在左邊的樹形視圖裡選擇StdAfx.cpp.如圖:(圖2)
這時原來的Project Option變成了 Source File Option(原來是工程,現在是一個檔案,當然變了)。在這裡我們可以看到 /Yc開關,/Yc的作用就是指定這個檔案來建立一個Pch檔案。/Yc後面的檔名是那個包含了穩定程式碼的標頭檔案,一個工程裡只能有一個檔案的可以有YC開關。VC就根據這個選項把 StdAfx.cpp編譯成一個Obj檔案和一個PCH檔案。
   然後我們再選擇一個其它的檔案來看看,如圖:
在這裡,Precomplier 選擇了 Use ………一項,標頭檔案是我們指定建立PCH 檔案的stdafx.h
檔案。事實上,這裡是使用工程裡的設定,(如圖1)/Yu”stdafx.h”。
   這樣,我們就設定好了預編譯標頭檔案。也就是說,我們可以使用預編譯頭功能了。以下是注意事項:
1):如果使用了/Yu,就是說使用了預編譯,我們在每個.cpp檔案的最開頭,我強調一遍是最開頭,包含 你指定產生pch檔案的.h檔案(預設是stdafx.h)不然就會有問題。如果你沒有包含這個檔案,就告訴你Unexpected file end. 如果你不是在最開頭包含的,你自己試以下就知道了,絕對有很驚人的效果…..
2)如果你把pch檔案不小心丟了,根據以上的分析,你只要讓編譯器生成一個pch檔案就可以了。也就是說把 stdafx.cpp(即指定/Yc的那個cpp檔案)從新編譯一遍就可以了。當然你可以傻傻的 Rebuild all。簡單一點就是選擇那個cpp檔案,按一下Ctrl + F7就可以了。
方法二。自動使用
很簡單隻要指定/YX就可以了。或者在上圖中選擇Automatic………就可以了。注意的事情是如果你指定了/Yc /Yu的話,/Yx是會被忽略的。前者的優先級別高一些。

相關推薦

為什麼編譯中加__OBJC__?

    因為在一個OC工程中,可能包含.m、.mm、.c、.cpp四類編譯檔案,這四類檔案均會引用.pch預編譯頭。       在編譯.c,.cpp時,因為語法不相容OC,所以預編譯頭中不能包含objc程式碼。      因為.pch是2類原始檔共用的,所以在pch

編譯 #include"編譯檔案"為何放在第一句

vs2010 預編譯頭 jiese1990 預編譯頭原理 你有兩個檔案a.cpp和b.cpp,都包含了同一個標頭檔案c.h。那麼正常的流程是:將c.h和a.cpp合併,編譯成a.obj;將c.h和b

為什麼使用編譯

許多初學 VC 的朋友也許都為那麼一個問題困擾過:    為什麼所有的 cpp 都必須 #include “stdafx.h”    也許請教了別的高手之後,他們會告訴你,這是預編譯頭,必須包含。可是,這到底是為什麼呢?預編譯頭有什麼用呢?    這得從標頭檔案的編譯原理講起

VC++ 使用編譯

radius 而已 tools filters mpi 鏈接 res 普通 mov 一、使用默認的預編譯頭 要使用預編譯頭,我們必須指定一個頭文件,這個頭文件包含我們不會經常改變的代碼和其他的頭文件,然後我們用這個頭文件來生成一個預編譯頭文件(.pch文件),想

#include”* .h“ 在查找編譯使用時跳過

ios warning 跳過 ima stream bsp 分享 strong ges warning C4627: “#include <windows.h>”: 在查找預編譯頭使用時跳過 解決辦法: 原因是沒有在cpp文件

error C1853: “DebugBigBuffer.pch”編譯文件來自編譯器的早期版本,或者編譯為 C++ 而在 C 中使用它(或相反)

view height 編譯 ont 彈出 對話 ngs 編譯器 -c <pre id="best-content-1299104064" mb-10"="" style="font-size: 14px; line-height: 28px; ">該錯誤是因為

初入 OpenGL ---白屏問題 -- glad.c在查找編譯遇到意外的文件結尾,是否忘記向源中添加#include "stdafx.h" ?

setw 檢查 wid event buffers cout stream swap turn 學習地址:https://learnopengl-cn.github.io 學習成果: 前言: 跟著教程走,用VS2017 配置完了OpenGL的運行.

C++---使用VS在C++程式設計中出現 fatal error C1010: 在查詢編譯時遇到意外的檔案結尾。是否忘記了向源中新增“#include "stdafx.h"”?

啦啦啦,好久沒寫部落格啦... 對於C++初學者來說適應一個新的編譯器還是需要蠻長一段時間的,現在我就給你們說說標題所說的這個問題吧... 第一步:選單--〉專案--〉設定,出現“專案設定”對話方塊,左邊展開專案,在“原始檔”中找到出錯的檔案。 第二步:在右邊選擇“C/C++”屬性頁,在Category

vs錯誤描述:fatal error C1010:在查詢編譯時遇到意外的檔案結尾。是否忘記了向源中新增“

錯誤描述:fatal error C1010:在查詢預編譯頭時遇到意外的檔案結尾。是否忘記了向源中新增“#include"stdafx.h"” 這個問題不一定是配置了使用預編譯頭造成的(專案-屬性-配

Visual Studio 有沒有在程式碼中關閉編譯選項的方法?

如題,不知道這樣說是不是清楚了。 就是說,我們把新的類引入我們自己的工程後, 如果我們的工程打開了預編譯頭,就需要在.cpp加上#include <stdafx.h>,或者關閉本工程或者那個cpp的預編譯頭選項;如果我們的工程關閉了預編譯頭,就要確保那個.cpp裡沒有#include

ZZ Cmake 設定編譯

專案採用cmake管理程式碼,涉及預編譯頭修改,在某某網下載了一個,寫的不錯,驗證過VC工程設定有效, GCC的哪位大仙驗證過共享一下唄。  # 建立預編譯頭 # Target是用來生成預編譯頭的專案Target;PrecompiledHeader和Precompiled

C/C++編譯的概念

預編譯頭的概念: 所謂的預編譯頭就是把一個工程中的那一部分程式碼,預先編譯好放在一個檔案裡(通常是 以.pch為副檔名的),這個檔案就稱為預編譯標頭檔案這些預先編譯好的程式碼可以是任何的 C/C++程式碼--------甚至是inline的函式,但是必須是穩定的,

未定義巨集或在編譯使用後定義發生改變

問題:在執行時提示類似 warning C4603: “SQ”: 未定義巨集或在預編譯頭使用後定義發生改變 一類的異常。 解決方法:調整以下程式碼前面的順序 #define SQ(y) ((y)*(y)) //定義帶引數的巨集 #include "stdafx.h" #inc

C++中編譯/yc /yu的區別

建立了一個空的工程,不停的新增新項,新增新類,數量越來越多,發現,每次修改一處,都會把所有的重新編譯,速度實在太慢,原來是沒有使用預編譯頭。我把別的工程的stdafx.h和stdafx.cpp拷過來,工程屬性改為"使用預編譯頭(/Yu)",編譯報錯,改為“建立預編譯頭(/Y

fatal error C1010: 在查詢編譯時遇到意外的檔案結尾。是否忘記了向源中新增“#include "stdafx.h

解決方式: 一. 1) 在解決方案資源管理器中,右擊相應的.cpp檔案,點選“屬性” 2) 在左側配置屬性中,點開“C/C++”,單擊“預編譯頭” 3) 更改右側第一行的“建立/使用預編譯頭”,把選項從“使用預編譯頭(/Yu)”改成“不使用預編譯頭” 4) 注:每一個報錯的.cp

VS中c++檔案呼叫c 函式 ,fatal error C1853 編譯檔案來自編譯器的早期版本,或者編譯為 C++ 而在 C 中使用它(或相反)

出現錯誤:error C1853: “Debug\ConsoleApplication1.pch”預編譯標頭檔案來自編譯器的早期版本,或者預編譯頭為 C++ 而在 C 中使用它(或相反) 相關資料:

gch檔案之淺談GCC編譯技術 收藏

其 實剛開始程式設計的時候,我是絲毫不重視編譯速度之類的問題的,原因很簡單,因為那時我用BASICA。後來一直用到C++ Builder,儘管Borland的廣告無時無刻不在吹噓其編譯速度,我卻從沒有對這個問題上心過,因為心裡根本沒有“編譯速度慢”這種概念。沒有壞, 哪來好?

解決fatal error C1859: “Debug\thread5.pch”意外的編譯錯誤的方法

VS2008中使用C++工程嚮導建立專案,直接編譯時出現該錯誤。有兩個方法來解決該問題: (1)clean專案,重新生成專案 (2)刪除該專案對應資料夾下的debug檔案中所有檔案,重新執行(1) 如果(1)有效,則無須執行(2);如果無效,則執行(2)

fatal error C1853 編譯檔案來自編譯器的早期版本,或者編譯為 C++ 而在 C 中使用它(或相反)

當 Visual C++ 專案啟用了預編譯頭 (Precompiled header) 功能時,如果專案中同時混合有 .c 和 .cpp 原始檔,則可能收到 C1853 編譯器錯誤:fatal error C1853: 'pjtname.pch' precompiled header file is fro

請問: "怎樣使用vs2010將指令新增到“StdAfx.h”或重新生成編譯"????

程式設計時出現了 Build started: Project: 0309aa, Configuration: Debug Win32 ------ 1>Build started 2011/3/9 16:35:14. 1>initialize build st