1.C運行庫和C語言函數庫(象stdio.h,stdlib.h等)是不是同一個東西?
2.在技術參考書中老說“Microsoft C運行庫”,那C運行庫到底是語言本身的庫還是作業系統的庫?其它作業系統如unix/linux中有沒有C運行庫?
3.為什麼既用Win32 API,還要用C運行庫?前者能替代後者嗎?
4.tchar.h、wchar.h是C運行庫的標頭檔還是API的標頭檔?TCHAR、WCHAR類型及_TEXT、TEXT等宏分別在哪個裡面定義的?
轉貼)
1)執行階段程式庫就是 C run-time library,是 C 而非 C++ 語言世界的概念:取這個名字就是因為你的 C 程式運行時需要這些庫中的函數.
2)C 語言是所謂的“小核心”語言,就其語言本身來說很小(不多的關鍵字,程式流程式控制制,資料類型等);所以,C 語言核心開發出來之後,Dennis Ritchie 和 Brian Kernighan 就用 C 本身重寫了 90% 以上的 UNIX 系統函數,並且把其中最常用的部分獨立出來,形成標頭檔和對應的 LIBRARY,C run-time library 就是這樣形成的。
3)隨後,隨著 C 語言的流行,各個 C 編譯器的生產商/個體/團體都遵循老的傳統,在不同平台上都有相對應的 Standard Library,但大部分實現都是與各個平台有關的。由於各個 C 編譯器對 C 的支援和理解有很多分歧和微妙的差別,所以就有了 ANSI C;ANSI C (主觀意圖上)詳細的規定了 C 語言各個要素的具體含義和編譯器實現要求,引進了新的函式宣告方式,同時訂立了 Standard Library 的標準形式。所以C執行階段程式庫由編譯器生產商提供。至於由其他廠商/個人/團體提供的標頭檔和庫函數,應當稱為第三方 C 運行庫(Third party C run-time libraries)。
4)C run-time library裡面含有初始化代碼,還有錯誤處理代碼(例如divide by zero處理)。你寫的程式可以沒有math庫,程式照樣運行,只是不能處理複雜的數學運算,不過如果沒有了C run-time庫,main()就不會被調用,exit()也不能被響應。因為C run-time library包含了C程式啟動並執行最基本和最常用的函數。
5)到了 C++ 世界裡,有另外一個概念:Standard C++ Library,它包括了上面所說的 C run-time library 和 STL。包含 C run-time library 的原因很明顯,C++ 是 C 的超集,沒有理由再重新來一個 C++ run-time library. VC針對C++ 加入的Standard C++ Library主要包括:LIBCP.LIB, LIBCPMT.LIB和 MSVCPRT.LIB
6)Windows環境下,VC提供的 C run-time library又分為動態執行階段程式庫和靜態執行階段程式庫。
動態執行階段程式庫主要是DLL庫檔案msvcrt.dll(or MSVCRTD.DLL for debug build),對應的Import library檔案是MSVCRT.LIB(MSVCRTD.LIB for debug build)
靜態執行階段程式庫(release版)對應的主要檔案是:
LIBC.LIB (Single thread static library, retail version)
LIBCMT.LIB (Multithread static library, retail version)
msvcrt.dll提供幾千個C函數,即使是像printf這麼低級的函數都在msvcrt.dll裡。其實你的程式運行時,很大一部分時間時在這些運行庫裡運行。在你的程式(release版)被編譯時間,VC會根據你的編譯選項(單線程、多線程或DLL)自動將相應的執行階段程式庫檔案(libc.lib,libcmt.lib或Import library msvcrt.lib)連結進來。
編譯時間到底哪個C run-time library聯入你的程式取決於編譯選項:
/MD, /ML, /MT, /LD (Use Run-Time Library)
你可以VC中通過以下方法設定選擇哪個C run-time library聯入你的程式:
To find these options in the development environment, click Settings on the Project menu. Then click the C/C++ tab, and click Code Generation in the Category box. See the Use Run-Time Library drop-down box.
從程式可移植性考慮,如果兩函數都可完成一種功能,選執行階段程式庫函數好,因為各個 C 編譯器的生產商對標準C Run-time library提供了統一的支援.
對該文的評論
han012 ( 2001-09-03)
很高興有許多網友參與討論,並對本文提出不同的觀點, 在此表示感謝.
另外針對一些網友的評論,我想有必要進行一些補充:
1)在Windows環境中,直到VC++4.0之前(如果我沒有記錯)只有靜態C run-time library,之後才有了動態C run-time library. 動態C run-time library是以DLL形式給出的. 從理論上說,不論是靜態還是動態C run-time library都是一些編譯好的,可以被啟動並執行二進位代碼. 所以從理論上說,只要符合調用規則,"同一作業系統平台上的不同編譯系統應該可以共用庫檔案"。在這一點上我也贊同PingPingPangPang. 正如在本文中沒有一處說 "C run-time library"只能被C/C++使用."
2)雖然從理論上說,C run-time library可以被其他編譯系統共用,但不得不承認C run-time library還是和C語言有更為密切的關係. 這就好比做衣服,C run-time library是為C語言量身定做的,別人或許可以穿,但大小未必合適. 從曆史上看,ANSI C是針對C語言制定了C Standard Library的標準形式(包括函數名,調用規則,參數等).即使別的語言可以使用,也必須尊從C run-time library的調用規則. 另外下面這段代碼對C run-time library函數fopen,printf的調用是最自然最直接的,雖然我不熟悉PASCAL,FORTRAN,但我相信如果它們希望調用C run-time library函數fopen,printf,恐怕還要多做一些工作.
FILE *stream;
... ...
if( (stream = fopen( "data", "r" )) == NULL )
printf( "The file 'data' was not opened/n" );
else
printf( "The file 'data' was opened/n" );
... ...
3)在Windows環境中,一個應用被載入時要經過以下步驟(截取部分說明,請忽略序號,只參照包含C執行階段程式庫的說明):
... ...
6.主線程為每個DLL調用_DLLMainCRTStartu() 函數。
當連結DLL時,連結器在產生的DLL檔案映象中嵌入了DLL的進入/退出函數的地址。預設時為,連結器假設進入函數名為:_DLLMainCRTStartup。該函數包含在C run-time library(C執行階段程式庫)檔案中。
當DLL檔案被映射到進程地址空間時,系統實際調用的是該函數_DLLMainCRTStartup, 而不是你的DLLMain函數。該函數執行:
a.初始化C run-time library(C執行階段程式庫)。
// Do runtime startup initializers.
_initterm( &__xi_a, &__xi_z );
... ...
7.主線程根據EXE檔案中的子系統值GUI/(CUI),執行WinMainCRTStartup/(MainCRTStartup)
a.得到一個新進程的全部命令列的指標。
b.得到一個新進程的環境變數指標。
c.通過STDLIB.H來初始化能被應用訪問的C run-time library(C執行階段程式庫)全域變數。例如
_osver, _winmajor, _winminor, _winver, _argc, _argv, _environ.
// Do runtime startup initializers.
_initterm( __xi_a, __xi_z );
... ...
所以說C run-time library(C執行階段程式庫)中確實包含了針對C run-time library自己的初始化代碼,例如初始化C run-time library(C執行階段程式庫)全域變數_argc, _argv, _environ. 我不知道在PASCAL,FORTRAN程式中是否可以調用這些全域變數.
僅供參考