做者:施洪寶html
三、g++參數說明,java
四、連接器ld參數:linux
一、對於c或者c++項目而言, 咱們認爲單個c或者cpp文件是一個編譯單元, 經過編譯器(gcc, g++, clang, clang++)能夠生成編譯後的二進制文件。例如: 編譯file1.cpp, 能夠生成file1.o。對於單個編譯單元而言, 裏面會有一些符號, 例如函數名稱, 變量名稱, 類名。這些符號能夠分爲三類:ios
#ifndef _MATH_H_ #define _MATH_H_ int add(int a, int b); #endif
#include "math.h" int add(int a, int b){ return a + b; }
#include <iostream> #include "math.h" using namespace std; int main(int argc, char **argv){ int a = 100, b = 200; int result = add(a, b); cout << result << endl; }
四、生成可執行文件c++
Using built-in specs. COLLECT_GCC=g++ COLLECT_LTO_WRAPPER=/usr/libexec/gcc/x86_64-redhat-linux/4.8.5/lto-wrapper Target: x86_64-redhat-linux Configured with: ../configure --prefix=/usr --mandir=/usr/share/man --infodir=/usr/share/info --with-bugurl=http://bugzilla.redhat.com/bugzilla --enable-bootstrap --enable-shared --enable-threads=posix --enable-checking=release --with-system-zlib --enable-__cxa_atexit --disable-libunwind-exceptions --enable-gnu-unique-object --enable-linker-build-id --with-linker-hash-style=gnu --enable-languages=c,c++,objc,obj-c++,java,fortran,ada,go,lto --enable-plugin --enable-initfini-array --disable-libgcj --with-isl=/builddir/build/BUILD/gcc-4.8.5-20150702/obj-x86_64-redhat-linux/isl-install --with-cloog=/builddir/build/BUILD/gcc-4.8.5-20150702/obj-x86_64-redhat-linux/cloog-install --enable-gnu-indirect-function --with-tune=generic --with-arch_32=x86-64 --build=x86_64-redhat-linux Thread model: posix gcc version 4.8.5 20150623 (Red Hat 4.8.5-36) (GCC) COMPILER_PATH=/usr/libexec/gcc/x86_64-redhat-linux/4.8.5/:/usr/libexec/gcc/x86_64-redhat-linux/4.8.5/:/usr/libexec/gcc/x86_64-redhat-linux/:/usr/lib/gcc/x86_64-redhat-linux/4.8.5/:/usr/lib/gcc/x86_64-redhat-linux/ LIBRARY_PATH=/usr/lib/gcc/x86_64-redhat-linux/4.8.5/:/usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../../lib64/:/lib/../lib64/:/usr/lib/../lib64/:/usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../:/lib/:/usr/lib/ COLLECT_GCC_OPTIONS='-v' '-o' 'main' '-shared-libgcc' '-mtune=generic' '-march=x86-64' /usr/libexec/gcc/x86_64-redhat-linux/4.8.5/collect2 --build-id --no-add-needed --eh-frame-hdr --hash-style=gnu -m elf_x86_64 -dynamic-linker /lib64/ld-linux-x86-64.so.2 -o main /usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../../lib64/crt1.o /usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../../lib64/crti.o /usr/lib/gcc/x86_64-redhat-linux/4.8.5/crtbegin.o -L/usr/lib/gcc/x86_64-redhat-linux/4.8.5 -L/usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../../lib64 -L/lib/../lib64 -L/usr/lib/../lib64 -L/usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../.. main.o math.o -lstdc++ -lm -lgcc_s -lgcc -lc -lgcc_s -lgcc /usr/lib/gcc/x86_64-redhat-linux/4.8.5/crtend.o /usr/lib/gcc/x86_64-redhat-linux/4.8.5/../../../../lib64/crtn.o
經過上面的介紹, 咱們知道一個c/cpp文件經過編譯連接, 最終生成可執行文件。不管任何語言, 程序員在寫代碼時, 都不可避免須要使用到庫, 本文主要介紹C/C++中的庫, 整體而言, 咱們將這些庫分爲靜態連接庫(一般以.a結尾),動態連接庫(一般以.so結尾)。首先咱們來看幾個問題:程序員
本節以hello world爲例,shell
#include <iostream> using namespace std; int main(int argc, char **argv){ cout << "hello world" << endl; }
linux-vdso.so.1 => (0x00007ffcf53fa000) libstdc++.so.6 => /lib64/libstdc++.so.6 (0x00007f7828b3b000) libm.so.6 => /lib64/libm.so.6 (0x00007f7828839000) libgcc_s.so.1 => /lib64/libgcc_s.so.1 (0x00007f7828623000) libc.so.6 => /lib64/libc.so.6 (0x00007f7828256000) /lib64/ld-linux-x86-64.so.2 (0x00007f7828e42000)
一、編譯: g++ -std=c++11 -fPIC -c math.cppbootstrap
三、使用這個靜態連接庫:windows
二、使用動態連接庫:app
./main: error while loading shared libraries: libmath.so: cannot open shared object file: No such file or directory
linux-vdso.so.1 => (0x00007ffd2adde000) libmath.so => not found libstdc++.so.6 => /lib64/libstdc++.so.6 (0x00007fd3b7ee6000) libm.so.6 => /lib64/libm.so.6 (0x00007fd3b7be4000) libgcc_s.so.1 => /lib64/libgcc_s.so.1 (0x00007fd3b79ce000) libc.so.6 => /lib64/libc.so.6 (0x00007fd3b7601000) /lib64/ld-linux-x86-64.so.2 (0x00007fd3b81ed000)
很奇怪, libmath.so沒有找到, 咱們在第三步編譯時明明將這個庫加入進去了。這個是因爲, 在連接階段, 連接器能夠在當前目錄找到libmath.so。執行階段, 搜索動態連接庫時, 並無包含當前目錄, 因此報錯。咱們能夠經過export LD_LIBRARY_PATH=/libpath將libmath.so所在路徑放入動態連接庫的搜索路徑中。此時便可成功執行。
從靜態連接庫生成的過程來看, 其本質就是將多個編譯單元(.o文件), 打包爲一個新的文件。連接靜態連接庫時, 會將靜態連接庫的代碼合併進程序中。
#ifndef __FIRST_H_ #define __FIRST_H_ #include <cstdio> void first(); #endif
#include"first.h" void first() { printf("This is first!\n"); }
#ifndef __SECOND_H_ #define __SECOND_H_ #include <cstdio> void second(); #endif
#include"first.h" #include"second.h" void second() { printf("This is second!\n"); first(); }
#include"second.h" int main() { second(); return 0; }
g++ -std=c++11 -fPIC -c first.cpp ar -crv libfirst.a first.o
g++ -std=c++11 -c second.cpp -L. -lfirst ar -crv libsecond.a second.o
執行: g++ -std=c++11 main.cpp -L. -lsecond -o main
會出現如下錯誤:
./libsecond.a(second.o): In function second()': second.cpp:(.text+0xf): undefined reference tofirst()’ collect2: error: ld returned 1 exit status
四、解釋說明
五、解決方案
同時連接libsecond.a, libfirst.a
g++ -std=c++11 second.cpp -fPIC -shared -o libsecond.so -L. -lfirst
g++ -std=c++11 main.cpp -L. -lsecond -o main
g++ -std=c++11 first.cpp -shared -fPIC -o libfirst.so
g++ -std=c++11 -c second.cpp -fPIC -L. -lfirst ar crv libsecond.a second.o
nm -u libsecond.a, 能夠看到_Z5firstv, 說明並無將libfirst.so中包含進libsecond.a
g++ -std=c++11 main.cpp -L. -lsecond -lfirst -o main
若是沒有連接first, 會發現連接錯誤, 找不到first函數的定義
g++ -std=c++11 first.cpp -shared -fPIC -o libfirst.so
g++ -std=c++11 second.cpp -shared -fPIC -o libsecond.so -L. -lfirst
nm -u libsecond.so, 能夠看到_Z5firstv, 這個就是first函數
ldd libsecond.so, 也能夠看到libfirst.so
能夠看出, 使用libsecond.so時, 仍然須要libfirst.so
g++ -std=c++11 main.cpp -L. -lsecond -o main
能夠看出, 可以成功編譯。
以前講過libsecond.so須要依賴libfirst.so, 此處爲什麼咱們只連接libsecond.so也能成功呢?這裏是由於連接器會自動搜索動態連接庫的依賴庫
一、動態庫升級問題?假設如今有2個程序: p1, p2, 一個動態連接庫libmath.so.1。若是如今math庫提供了新版本libmath.so.2, 程序p1須要使用libmath.so.2的新功能, p2則不想使用, 此時該如何升級math庫?
二、某個動態連接庫lib1動態連接了庫libbase, 如今應用程序中使用了lib1以及libbase, 編譯應用程序時, 是否須要連接libbase?
三、菱形依賴問題, A依賴於B以及C, B、C都依賴於D, 可是是不一樣版本, 例如B依賴於D1, C依賴於D2, 這種狀況下如何連接?
連接器的參數, 直接連接兩個版本。ld的參數–default-symver或者–version-script
四、討論
http://blog.chinaunix.net/uid...
https://www.cnblogs.com/fnlin...
https://blog.csdn.net/coolwat...
https://blog.habets.se/2012/0...