1. 程式人生 > >C/C++—— #include“stdafx.h”詳解

C/C++—— #include“stdafx.h”詳解

再選【下一步】-預設情況下,我們都是在【空專案】前打上對勾,建立一個純的C或C++程式。

我們這裡選擇預設,即【預編譯頭】前打上對勾,之後【完成】

2,在Test專案中,標頭檔案有stdafx.h和targetver.h,原始檔有stdafx.cpp和Test.cpp,還有一個ReadMe.txt檔案。

3,預設的主函式為

#include "stdafx.h"  


int _tmain(int argc, _TCHAR* argv[])  
{  
    return 0;  
}  

和我們一般的寫的程式不一樣,我們一般會寫成下面的形式:

#include  <stdio.h>  
int main() { return 0; }

4,我們要糾結,這裡怎麼沒有了stdio.h了呢,取而代之的是stdafx.h呢?
我們開啟stdafx.h檔案,如下:

// stdafx.h : 標準系統包含檔案的包含檔案,  
// 或是經常使用但不常更改的  
// 特定於專案的包含檔案  
//  

#pragma once  

#include "targetver.h"  

#include <stdio.h>  
#include <tchar.h>  



// TODO: 在此處引用程式需要的其他標頭檔案 

可以看到,這裡包含了stdio.h庫檔案,為什麼包含到這裡呢?我們先不急著解答。
先看看,stdafx.cpp檔案:

// stdafx.cpp : 只包括標準包含檔案的原始檔  
// Test.pch 將作為預編譯頭  
// stdafx.obj 將包含預編譯型別資訊  

#include "stdafx.h"  

// TODO: 在 STDAFX.H 中  
// 引用任何所需的附加標頭檔案,而不是在此檔案中引用

這裡面只有包含的庫檔案,沒有操作,它是做什麼用的呢?
在ReadMe.txt中,我們看到這麼一段話:

其他標準檔案:
StdAfx.h, StdAfx.cpp
這些檔案用於生成名為 Test.pch 的預編譯頭 (PCH) 檔案和名為 StdAfx.obj 的預編譯型別檔案。

原來是生成預編譯標頭檔案!

5,百科上參看下:

一、什麼是預編譯頭?   
  所謂預編譯頭,就是把標頭檔案事先編譯成一種二進位制的中間格式,供後續的編譯過程使用。預編譯頭物理上與通常的的.obj檔案是一樣的,但是千萬不要把這個中間格式與. o/.obj/.a/.lib的格式混淆,他們是截然不同的!所以預編譯標頭檔案的特性和目標檔案也不同(儘管他們都屬於某種中間檔案)。編譯入預編譯頭的.h,.c,.cpp檔案在整個編譯過程中,只編譯一次,如預編譯頭所涉及的部分不發生改變的話,在隨後的編譯過程中此部分不重新進行編譯。進而大大提高編譯速度,並便於對標頭檔案進行管理,也有助於杜絕重複包含問題。——但也有類似的地方的,比如,它們都是編譯器之間不相容的^_^,就是說你不能把VC生成的預編譯頭拿到GCC上去用。甚至副檔名都不一樣,VC的是大家都熟悉的. pch,而GCC的,是.gch。   
二、什麼時候使用預編譯頭?   
  當大多.c.cpp檔案都需要相同的標頭檔案時。   
  當某些程式碼被大量重複使用時。   
  當匯入某些不同庫都有實現的函式,併產生混亂時。  

6,可以知道預編譯,是為了提高編譯速度!
再看下面一段話

stdafx.h    

1名稱的英文全稱為:Standard Application Fram Extend  
  所謂標頭檔案預編譯,就是把一個工程(Project)中使用的一些MFC標準標頭檔案(如Windows.H、Afxwin.H)預先編譯,以後該工程編譯時,不再編譯這部分標頭檔案,僅僅使用預編譯的結果。這樣可以加快編譯速度,節省時間。   
  預編譯標頭檔案通過編譯stdafx.cpp生成,以工程名命名,由於預編譯的標頭檔案的字尾是“pch”,所以編譯結果檔案是projectname.pch。   
  編譯器通過一個頭檔案stdafx.h來使用預編譯標頭檔案。stdafx.h這個標頭檔案名是可以在project的編譯設定裡指定的。編譯器認為,所有在指令#include "stdafx.h"前的程式碼都是預編譯的,它跳過#include "stdafx. h"指令,使用projectname.pch編譯這條指令之後的所有程式碼。   
  因此,所有的MFC實現檔案第一條語句都是:#include "stdafx.h"。  

2 詳細工作原理及作用  
  stdafx.h中沒有函式庫,只是定義了一些環境引數,使得編譯出來的程式能在32位的作業系統環境下執行。   
  Windows和MFC的include檔案都非常大,即使有一個快速的處理程式,編譯程式也要花費相當長的時間來完成工作。由於每個.CPP檔案都包含相同的include檔案,為每個.CPP檔案都重複處理這些檔案就顯得很傻了。   
  為避免這種浪費,AppWizard和VisualC++編譯程式一起進行工作,如下所示:   
  ◎AppWizard建立了檔案stdafx.h,該檔案包含了所有當前工程檔案需要的MFCinclude檔案。且這一檔案可以隨被選擇的選項而變化。   
  ◎AppWizard然後就建立stdafx.cpp。這個檔案通常都是一樣的。   
  ◎然後AppWizard就建立起工程檔案,這樣第一個被編譯的檔案就是stdafx.cpp。   
  ◎當VisualC++編譯stdafx.cpp檔案時,它將結果儲存在一個名為stdafx.pch的檔案裡。(副檔名pch表示預編譯標頭檔案。)   
  ◎當VisualC++編譯隨後的每個.cpp檔案時,它閱讀並使用它剛生成的.pch檔案。VisualC++不再分析Windowsinclude檔案,除非你又編輯了stdafx.cpp或stdafx.h。   
  這個技術很精巧,你不這麼認為嗎?(還要說一句,Microsoft並非是首先採用這種技術的公司,Borland才是。)在這個過程中你必須遵守以下規則:   
  ◎你編寫的任何.cpp檔案都必須首先包含stdafx.h。   
  ◎如果你有工程檔案裡的大多數.cpp檔案需要.h檔案,順便將它們加在stdafx.h(後部)上,然後預編譯stdafx.cpp。   
  ◎由於.pch檔案具有大量的符號資訊,它是你的工程檔案裡最大的檔案。   
  如果你的磁碟空間有限,你就希望能將這個你從沒使用過的工程檔案中的.pch檔案刪除。執行程式時並不需要它們,且隨著工程檔案的重新建立,它們也自動地重新建立。  

要點:
(1). Windows和MFC的include檔案都非常大,即使有一個快速的處理程式,編譯程式也要花費相當長的時間來完成工作。由於每個.CPP檔案都包含相同的include檔案,為每個.CPP檔案都重複處理這些檔案就顯得很傻了。
(2).MFC中,你編寫的任何.cpp檔案都必須首先包含stdafx.h。

(3).由於.pch檔案具有大量的符號資訊,它是你的工程檔案裡最大的檔案,不需要的時候可以刪除。

7,我們可以自己設定自己的預編譯檔案,不一定非要是stdafx.h

右擊stdafx.cpp檔案,選擇【屬性】-【C/C++】-【預編譯頭】-【建立/使用預編譯頭】

可以看到有3個選項:【建立預編譯頭】、【使用預編譯頭】和【不使用預編譯頭】。

stdafx.cpp肯定選擇的是【建立預編譯頭】

============================================================================

再右擊Test.cpp檔案,選擇【屬性】-【C/C++】-【預編譯頭】-【建立/使用預編譯頭】

可見,它選擇的是【使用預編譯頭】,下面選的檔案是StdAfx.h檔案,最下面指明的是pch檔案的路徑

================================================================================

如何建立自己的預編譯檔案,其實過程正如上面我們講的過程。
如果工程很大,標頭檔案很多,而有幾個標頭檔案又是經常要用的,那麼

1,把這些標頭檔案全部寫到一個頭檔案裡面去,比如寫到preh.h

2,寫一個preh.c,裡面只一句話:#include “preh.h” ,為了生存pch預編譯檔案

3,對於preh.c,在【屬性】裡面設定【建立預編譯頭】,對於其他.c檔案,設定【使用預編譯頭】