關鍵點: 1. #include <...> 不會搜索當前目錄java
2. 使用 -I 參數指定的頭文件路徑僅次於 搜索當前路徑。linux
3. gcc -E -v 能夠輸出頭文件路徑搜索過程c++
C++編譯時,教科書中寫道:#include 「headfile.h」優先在當前目錄查找頭文件;#include < headfile.h >從系統默認路徑查找頭文件。先前覺得系統默認路徑是環境變量$PATH指定的路徑,在系統上一查,傻了眼:bash
-bash-3.2$ echo $PATHjvm
/usr/local/bin:/bin:/usr/bin:/sbin:/usr/sbin:/usr/X11R6/bin:/usr/java/j2re1.4.0/bin:/usr/atria/bin:/ccase/bin:/home/devcomp/binui
全是bin目錄,$PATH是運行可執行文件時的搜索路徑,與include頭文件的搜索路徑無關,可能很多人犯了我這樣的錯誤。get
頭文件:it
1. #include 「headfile.h」io
搜索順序爲:編譯
①先搜索當前目錄
②而後搜索-I指定的目錄
③再搜索gcc的環境變量CPLUS_INCLUDE_PATH(C程序使用的是C_INCLUDE_PATH)
④最後搜索gcc的內定目錄
/usr/include
/usr/local/include
/usr/lib/gcc/x86_64-redhat-linux/4.1.1/include
各目錄存在相同文件時,先找到哪一個使用哪一個。
2. #include <headfile.h>
①先搜索-I指定的目錄
②而後搜索gcc的環境變量CPLUS_INCLUDE_PATH
③最後搜索gcc的內定目錄
/usr/include
/usr/local/include
/usr/lib/gcc/x86_64-redhat-linux/4.1.1/include
與上面的相同,各目錄存在相同文件時,先找到哪一個使用哪一個。這裏要注意,#include<>方式不會搜索當前目錄!
這裏要說下include的內定目錄,它不是由$PATH環境變量指定的,而是由g++的配置prefix指定的(知道它在安裝g++時能夠指定,不知安裝後如何修改的,多是修改配置文件,須要時再研究下):
-bash-3.2$ g++ -v
Using built-in specs.
Target: x86_64-redhat-linux
Configured with: ../configure --prefix=/usr --mandir=/usr/share/man --infodir=/usr/share/info --enable-shared --enable-threads=posix --enable-checking=release --with-system-zlib --enable-__cxa_atexit --disable-libunwind-exceptions --enable-libgcj-multifile --enable-languages=c,c++,objc,obj-c++,java,fortran,ada --enable-java-awt=gtk --disable-dssi --enable-plugin --with-java-home=/usr/lib/jvm/java-1.4.2-gcj-1.4.2.0/jre --with-cpu=generic --host=x86_64-redhat-linux
Thread model: posix
gcc version 4.1.2 20080704 (Red Hat 4.1.2-46)
在安裝g++時,指定了prefix,那麼內定搜索目錄就是:
Prefix/include
Prefix/local/include
Prefix/lib/gcc/--host/--version/include
編譯時能夠經過-nostdinc++選項屏蔽對內定目錄搜索頭文件。
庫文件:
編譯的時候:
①gcc會去找-L
②再找gcc的環境變量LIBRARY_PATH
③再找內定目錄 /lib /usr/lib /usr/local/lib 這是當初compile gcc時寫在程序內的(不可配置的?)
運行時動態庫的搜索路徑:
動態庫的搜索路徑搜索的前後順序是:
①編譯目標代碼時指定的動態庫搜索路徑(這是經過gcc 的參數"-Wl,-rpath,"指定。當指定多個動態庫搜索路徑時,路徑之間用冒號":"分隔)
②環境變量LD_LIBRARY_PATH指定的動態庫搜索路徑(當經過該環境變量指定多個動態庫搜索路徑時,路徑之間用冒號":"分隔)
③配置文件/etc/ld.so.conf中指定的動態庫搜索路徑;
④默認的動態庫搜索路徑/lib;
⑤默認的動態庫搜索路徑/usr/lib。
(應注意動態庫搜尋路徑並不包括當前文件夾,因此立即使可執行文件和其所需的so文件在同一文件夾,也會出現找不到so的問題,類同#include <header_file>不搜索當前目錄)