Gcc详解以及静态库、动态库生成
[轉(zhuǎn)] Gcc詳解以及靜態(tài)庫(kù)、動(dòng)態(tài)庫(kù)生成
http://www.360doc.com/content/10/0619/14/1795182_33985297.shtml
1。gcc包含的c/c++編譯器
gcc,cc,c++,g++,gcc和cc是一樣的,c++和g++是一樣的,(沒(méi)有看太明白前面這半句是什
么意思:))一般c程序就用gcc編譯,c++程序就用g++編譯
2。gcc的基本用法
gcc test.c這樣將編譯出一個(gè)名為a.out的程序
gcc test.c -o test這樣將編譯出一個(gè)名為test的程序,-o參數(shù)用來(lái)指定生成程序的名
字
3。為什么會(huì)出現(xiàn)undefined reference to 'xxxxx'錯(cuò)誤?
首先這是鏈接錯(cuò)誤,不是編譯錯(cuò)誤,也就是說(shuō)如果只有這個(gè)錯(cuò)誤,說(shuō)明你的程序源碼本
身沒(méi)有問(wèn)題,是你用編譯器編譯時(shí)參數(shù)用得不對(duì),你沒(méi)
有指定鏈接程序要用到得庫(kù),比如你的程序里用到了一些數(shù)學(xué)函數(shù),那么你就要在編譯
參數(shù)里指定程序要鏈接數(shù)學(xué)庫(kù),方法是在編譯命令行里加入-lm。
4。-l參數(shù)和-L參數(shù)
-l參數(shù)就是用來(lái)指定程序要鏈接的庫(kù),-l參數(shù)緊接著就是庫(kù)名,那么庫(kù)名跟真正的庫(kù)文
件名有什么關(guān)系呢?
就拿數(shù)學(xué)庫(kù)來(lái)說(shuō),他的庫(kù)名是m,他的庫(kù)文件名是libm.so,很容易看出,把庫(kù)文件名的
頭lib和尾.so去掉就是庫(kù)名了。
好了現(xiàn)在我們知道怎么得到庫(kù)名了,比如我們自已要用到一個(gè)第三方提供的庫(kù)名字叫l(wèi)ib
test.so,那么我們只要把libtest.so拷貝到/usr/lib
里,編譯時(shí)加上-ltest參數(shù),我們就能用上libtest.so庫(kù)了(當(dāng)然要用libtest.so庫(kù)里
的函數(shù),我們還需要與libtest.so配套的頭文件)。
放在/lib和/usr/lib和/usr/local/lib里的庫(kù)直接用-l參數(shù)就能鏈接了,但如果庫(kù)文件
沒(méi)放在這三個(gè)目錄里,而是放在其他目錄里,這時(shí)我們
只用-l參數(shù)的話,鏈接還是會(huì)出錯(cuò),出錯(cuò)信息大概是:“/usr/bin/ld: cannot find
-lxxx”,也就是鏈接程序ld在那3個(gè)目錄里找不到
libxxx.so,這時(shí)另外一個(gè)參數(shù)-L就派上用場(chǎng)了,比如常用的X11的庫(kù),它放在/usr/X11R
6/lib目錄下,我們編譯時(shí)就要用-L/usr/X11R6/lib -
lX11參數(shù),-L參數(shù)跟著的是庫(kù)文件所在的目錄名。再比如我們把libtest.so放在/aaa/bb
b/ccc目錄下,那鏈接參數(shù)就是-L/aaa/bbb/ccc -ltest
另外,大部分libxxxx.so只是一個(gè)鏈接,以RH9為例,比如libm.so它鏈接到/lib/libm.s
o.x,/lib/libm.so.6又鏈接到/lib/libm-2.3.2.so,
如果沒(méi)有這樣的鏈接,還是會(huì)出錯(cuò),因?yàn)閘d只會(huì)找libxxxx.so,所以如果你要用到xxxx
庫(kù),而只有l(wèi)ibxxxx.so.x或者libxxxx-x.x.x.so,做一
個(gè)鏈接就可以了ln -s libxxxx-x.x.x.so libxxxx.so
手工來(lái)寫鏈接參數(shù)總是很麻煩的,還好很多庫(kù)開發(fā)包提供了生成鏈接參數(shù)的程序,名字
一般叫xxxx-config,一般放在/usr/bin目錄下,比如
gtk1.2的鏈接參數(shù)生成程序是gtk-config,執(zhí)行g(shù)tk-config --libs就能得到以下輸出"-
L/usr/lib -L/usr/X11R6/lib -lgtk -lgdk -rdynamic
-lgmodule -lglib -ldl -lXi -lXext -lX11 -lm",這就是編譯一個(gè)gtk1.2程序所需的g
tk鏈接參數(shù),xxx-config除了--libs參數(shù)外,還有一個(gè)參數(shù)是--cflags用來(lái)生成頭文
件包含目錄的,也就是-I參數(shù),在下面我們將會(huì)講到。你可以試試執(zhí)行g(shù)tk-config
--libs --cflags,看看輸出結(jié)果。
現(xiàn)在的問(wèn)題就是怎樣用這些輸出結(jié)果了,最笨的方法就是復(fù)制粘貼或者照抄,聰明的辦
法是在編譯命令行里加入這個(gè)`xxxx-config --libs --
cflags`,比如編譯一個(gè)gtk程序:gcc gtktest.c `gtk-config --libs --cflags`這樣
就差
不多了。注意`不是單引號(hào),而是1鍵左邊那個(gè)鍵。
除了xxx-config以外,現(xiàn)在新的開發(fā)包一般都用pkg-config來(lái)生成鏈接參數(shù),使用方法
跟xxx-config類似,但xxx-config是針對(duì)特定的開發(fā)包
,但pkg-config包含很多開發(fā)包的鏈接參數(shù)的生成,用pkg-config --list-all命令可以
列出所支持的所有開發(fā)包,pkg-config的用法就是pkg
-config pagName --libs --cflags,其中pagName是包名,是pkg-config--list-all里
列出名單中的一個(gè),比如gtk1.2的名字就是gtk+,pkg-
config gtk+ --libs --cflags的作用跟gtk-config --libs --cflags是一樣的。比如:
gcc gtktest.c `pkg-config gtk+ --libs --cflags`
。
5。-include和-I參數(shù)
-include用來(lái)包含頭文件,但一般情況下包含頭文件都在源碼里用#include xxxxxx實(shí)現(xiàn)
,-include參數(shù)很少用。-I參數(shù)是用來(lái)指定頭文件目錄
,/usr/include目錄一般是不用指定的,gcc知道去那里找,但是如果頭文件不在/usr/i
nclude里我們就要用-I參數(shù)指定了,比如頭文件放
在/myinclude目錄里,那編譯命令行就要加上-I/myinclude參數(shù)了,如果不加你會(huì)得到
一個(gè)"xxxx.h: No such file or directory"的錯(cuò)誤。-I
參數(shù)可以用相對(duì)路徑,比如頭文件在當(dāng)前目錄,可以用-I.來(lái)指定。上面我們提到的--cf
lags參數(shù)就是用來(lái)生成-I參數(shù)的。
6。-O參數(shù)
這是一個(gè)程序優(yōu)化參數(shù),一般用-O2就是,用來(lái)優(yōu)化程序用的,比如gcc test.c -O2,優(yōu)
化得到的程序比沒(méi)優(yōu)化的要小,執(zhí)行速度可能也有所提
高(我沒(méi)有測(cè)試過(guò))。
7。-shared參數(shù)
編譯動(dòng)態(tài)庫(kù)時(shí)要用到,比如gcc -shared test.c -o libtest.so
8。幾個(gè)相關(guān)的環(huán)境變量
PKG_CONFIG_PATH:用來(lái)指定pkg-config用到的pc文件的路徑,默認(rèn)是/usr/lib/pkgconf
ig,pc文件是文本文件,擴(kuò)展名是.pc,里面定義開發(fā)
包的安裝路徑,Libs參數(shù)和Cflags參數(shù)等等。
CC:用來(lái)指定c編譯器。
CXX:用來(lái)指定cxx編譯器。
LIBS:跟上面的--libs作用差不多。
CFLAGS:跟上面的--cflags作用差不多。
CC,CXX,LIBS,CFLAGS手動(dòng)編譯時(shí)一般用不上,在做configure時(shí)有時(shí)用到,一般情況
下不用管。
環(huán)境變量設(shè)定方法:export ENV_NAME=xxxxxxxxxxxxxxxxx
9。關(guān)于交叉編譯
交叉編譯通俗地講就是在一種平臺(tái)上編譯出能運(yùn)行在體系結(jié)構(gòu)不同的另一種平臺(tái)上,比
如在我們地PC平臺(tái)(X86 CPU)上編譯出能運(yùn)行在sparc
CPU平臺(tái)上的程序,編譯得到的程序在X86 CPU平臺(tái)上是不能運(yùn)行的,必須放到sparc
CPU平臺(tái)上才能運(yùn)行。
當(dāng)然兩個(gè)平臺(tái)用的都是linux。
這種方法在異平臺(tái)移植和嵌入式開發(fā)時(shí)用得非常普遍。
相對(duì)與交叉編譯,我們平常做的編譯就叫本地編譯,也就是在當(dāng)前平臺(tái)編譯,編譯得到
的程序也是在本地執(zhí)行。
用來(lái)編譯這種程序的編譯器就叫交叉編譯器,相對(duì)來(lái)說(shuō),用來(lái)做本地編譯的就叫本地編
譯器,一般用的都是gcc,但這種gcc跟本地的gcc編譯器
是不一樣的,需要在編譯gcc時(shí)用特定的configure參數(shù)才能得到支持交叉編譯的gcc。
為了不跟本地編譯器混淆,交叉編譯器的名字一般都有前綴,比如sparc-xxxx-linux-gn
u-gcc,sparc-xxxx-linux-gnu-g++ 等等
10。交叉編譯器的使用方法
使用方法跟本地的gcc差不多,但有一點(diǎn)特殊的是:必須用-L和-I參數(shù)指定編譯器用spar
c系統(tǒng)的庫(kù)和頭文件,不能用本地(X86)
的庫(kù)(頭文件有時(shí)可以用本地的)。
例子:
sparc-xxxx-linux-gnu-gcc test.c -L/path/to/sparcLib -I/path/to/sparcInclude
Linux下生成動(dòng)態(tài)和靜態(tài)庫(kù)
在用c寫程序時(shí),很多時(shí)候需要存儲(chǔ)一些簡(jiǎn)單的數(shù)據(jù),如果為此而用mysql數(shù)據(jù)庫(kù)就有些大才小用了,可以把這些數(shù)據(jù)以結(jié)構(gòu)的形寫入文件,然后再需要時(shí)讀取文件,取出數(shù)據(jù)。
如下是定義函數(shù)的源文件和頭文件:
源文件struct.c:
#include "struct.h"
//第一個(gè)參數(shù)是要寫入的文件名,第二個(gè)參數(shù)是緩沖區(qū),第三個(gè)參數(shù)是緩沖區(qū)大小,第四個(gè)參數(shù)是打開文件流的形態(tài),返回TRUE表示寫入成功,返回FALSE表示寫入失敗
int writeStruct(const char *fileName,char *buffer,int bufferLen,char *mode){ int ret; FILE *fileID = NULL;
fileID = fopen(fileName,mode); if (fileID == NULL){ perror("fopen"); goto writeEnd; } rewind(fileID); ret = fwrite(buffer,bufferLen,1,fileID); if (ret <= 0){ perror("fwrite"); goto writeEnd; } if (fileID != NULL){ fclose(fileID); fileID = NULL; } return TRUE;
writeEnd: if (fileID != NULL){ fclose(fileID); fileID = NULL; } return FALSE; }
//第一個(gè)參數(shù)是要讀取的文件名,第二個(gè)參數(shù)是緩沖區(qū),第三個(gè)參數(shù)是緩沖區(qū)大小,第四個(gè)參數(shù)是打開文件流的形態(tài),返回TRUE表示讀取成功,返回FALSE表示讀取失敗
int readStruct(const char *fileName,char *buffer,int bufferLen,char *mode){ int ret; FILE *fileID = NULL;
fileID = fopen(fileName,mode); if (fileID == NULL){ perror("fopen"); goto readEnd; } rewind(fileID); memset(buffer,0,sizeof(buffer)); ret = fread(buffer,bufferLen,1,fileID); if (ret >= 0){ strcat(buffer,"\0"); }else{ perror("fread") ; goto readEnd; } if (fileID != NULL){ fclose(fileID); fileID = NULL; } return TRUE;
readEnd: if (fileID != NULL){ fclose(fileID); fileID = NULL; } return FALSE; }
頭文件struct.h:
#ifndef OWNSTRUCT_H_ #define OWNSTRUCT_H_
#include #include #include
#define FALSE 0 #define TRUE 1
//第一個(gè)參數(shù)是要寫入的文件名,第二個(gè)參數(shù)是緩沖區(qū),第三個(gè)參數(shù)是緩沖區(qū)大小,第四個(gè)參數(shù)是打開文件流的形態(tài),返回TRUE表示寫入成功,返回FALSE表示寫入失敗
int writeStruct(const char *fileName,char *buffer,int bufferLen,char *mode);
//第一個(gè)參數(shù)是要讀取的文件名,第二個(gè)參數(shù)是緩沖區(qū),第三個(gè)參數(shù)是緩沖區(qū)大小,第四個(gè)參數(shù)是打開文件流的形態(tài),返回TRUE表示讀取成功,返回FALSE表示讀取失敗
int readStruct(const char *fileName,char *buffer,int bufferLen,char *mode);
#endif
為了使用方便,可以把這兩個(gè)函數(shù)接口定義為動(dòng)態(tài)鏈接庫(kù)或靜態(tài)鏈接庫(kù)。用動(dòng)態(tài)鏈接庫(kù)編譯生成的可執(zhí)行文件需調(diào)用.so文件方可正常運(yùn)行,靈活但稍顯麻煩;用靜態(tài)鏈接庫(kù)編譯生成的可執(zhí)行文件可直接運(yùn)行,不用再調(diào)用如.so般的依賴庫(kù)文件,簡(jiǎn)單但不靈活。
靜態(tài)鏈接庫(kù):
1、編譯生成目標(biāo)文件
gcc -c struct.c
2、創(chuàng)建靜態(tài)庫(kù)
ar cqs libstruct.a struct.o (順序不能亂)
3、鏈接靜態(tài)鏈接庫(kù),生成可執(zhí)行文件
gcc main.c -static -L. -lstruct -o main
動(dòng)態(tài)鏈接庫(kù):
1、編譯成動(dòng)態(tài)鏈接庫(kù)
gcc struct.c -fPIC -shared -o libstruct.so
2、鏈接動(dòng)態(tài)鏈接庫(kù),生成可執(zhí)行文件
gcc main.c -L. -lstruct -o main
3、設(shè)置庫(kù)文件的環(huán)境路徑
1)在bashrc或profile文件里用LD_LIBRARY_PATH定義,然后用source加載。
2)把庫(kù)路徑添加到ld.so.conf文件中,然后用ldconfig加載。
3)ldconfig /home/user/lib,僅能暫時(shí)性使用,若下次ldconfig時(shí)此目錄下的動(dòng)態(tài)鏈接庫(kù)就不能被共享了。
gcc一些參數(shù)解析
-shared:指定生成動(dòng)態(tài)鏈接庫(kù)。
-static:指定生成靜態(tài)鏈接庫(kù)。
-fPIC:表示編譯為位置獨(dú)立的代碼,用于編譯共享庫(kù)。目標(biāo)文件需要?jiǎng)?chuàng)建成位置無(wú)關(guān)碼,概念上就是在可執(zhí)行程序裝載它們的時(shí)候,它們可以放在可執(zhí)行程序的內(nèi)存里的任何地方。
-L.:表示要連接的庫(kù)在當(dāng)前目錄中。
-l:指定鏈接時(shí)需要的動(dòng)態(tài)庫(kù)。編譯器查找動(dòng)態(tài)連接庫(kù)時(shí)有隱含的命名規(guī)則,即在給出的名字前面加上lib,后面加上.so來(lái)確定庫(kù)的名稱。
-Wall:生成所有警告信息。
-ggdb:此選項(xiàng)將盡可能的生成gdb的可以使用的調(diào)試信息。
-g:編譯器在編譯的時(shí)候產(chǎn)生調(diào)試信息。
-c:只激活預(yù)處理、編譯和匯編,也就是把程序做成目標(biāo)文件(.o文件)。
-Wl,options:把參數(shù)(options)傳遞給鏈接器ld。如果options中間有逗號(hào),就將options分成多個(gè)選項(xiàng),然后傳遞給鏈接程序。
實(shí)現(xiàn)例子:
fun.c:
1 #include "func.h"
? 2? int func(int n)
? 3? {
? 4? int sum=0,i;
? 5? for(i=0; i<n; i++)
? 6? {
? 7? sum+=i;
? 8? }
? 9? return sum;
?10? }
?
?test.c:
?
?
?1 #include <stdio.h>
?2 #include "func.h"
? main()
?15? {
?16? int i;
?17? long result = 0;
?18 long zhang=0;
?19? for(i=1; i<=100; i++)
?20? {
?21? result += i;
?22? }
?23
?24? printf("result[1-100] = %d \n", result );
?25
?26 zhang=func(250);
?27? printf("result[1-250] = %d \n", zhang );
?28
?29? }
一、
gcc func.c test.c
會(huì)有a.out文件
運(yùn)行./a.out
結(jié)果:
result[1-100] = 5050
result[1-250] = 31125
二、
gcc? fun.c test.c -o test
會(huì)有test
運(yùn)行:./test
結(jié)果:
result[1-100] = 5050
result[1-250] = 31125
三、生成靜態(tài)庫(kù):
1、gcc -c func.c
2、ar cqs libfunc.a func.o
3、gcc test.c -static -L. -lfunc-o test
運(yùn)行./test
結(jié)果:
result[1-100] = 5050
result[1-250] = 31125
四、生成動(dòng)態(tài)庫(kù)
1、gcc func.c -fPIC -shared -o libfunc.so
2、gcc test.c? -L. -lfunc -o test
3、export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:.
4、測(cè)試ldd test
5、運(yùn)行:./test
結(jié)果:
result[1-100] = 5050
result[1-250] = 31125
生存動(dòng)態(tài)庫(kù)的好處是,若是想改變func.c的算法的話,只要修改好func.c,重新生成.so文件,運(yùn)行./test便可,方便
總結(jié)
以上是生活随笔為你收集整理的Gcc详解以及静态库、动态库生成的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: c语言 周期
- 下一篇: 关于oracle sql语句查询时表名和