Java 程序員眼里的 Gcc

作為一名 Java 程序員,對(duì) gcc 的基本使用總是記不住,很不熟練,今天寫(xiě)篇文章用最簡(jiǎn)單的方式記錄下。
編譯的過(guò)程
寫(xiě)個(gè) hello.c 代碼:
#include <stdio.h>
#define STR "hello world\n"
void main() {
printf(STR);
}
第一步:預(yù)處理(preprocess)
其實(shí)就是以下三個(gè)動(dòng)作:
- 頭文件展開(kāi)
 - 宏定義直接替換
 - 條件編譯不成立就去掉
 
gcc -E hello.c -o hello.i
第二步:編譯(compile)
轉(zhuǎn)換成匯編語(yǔ)言:
gcc -S hello.i -o hello.s
第三步:匯編(assemble)
轉(zhuǎn)換成二進(jìn)制(ELF relocatable):
gcc -c hello.s -o hello.o
第四步:鏈接(link)
具體分為動(dòng)態(tài)鏈接和靜態(tài)鏈接:
# 動(dòng)態(tài)鏈接
gcc hello.o -o hello
# 靜態(tài)鏈接
gcc hello.o -o hello -static
靜態(tài)庫(kù)制作
寫(xiě)一個(gè)加法函數(shù):
int add(int a, int b) {
return a+b;
}
編譯成 .o:
gcc -c add.c -o add.o
制作成靜態(tài)庫(kù):
ar rcs libadd.a add.o
編寫(xiě)測(cè)試代碼:
#include <stdio.h>
void main(){
printf("%d", add(1,2));
}
編譯成可執(zhí)行文件:
#寫(xiě)法一
gcc test.c -o test libadd.a
#寫(xiě)法二
gcc test.c -o test -static -ladd -L ./
執(zhí)行:
./test
3
動(dòng)態(tài)庫(kù)制作
寫(xiě)一個(gè)加法函數(shù):
int add(int a, int b) {
return a+b;
}
編譯成 .o:
gcc -c add.c -o add.o -fPIC
制作成動(dòng)態(tài)庫(kù):
gcc -shared -o libadd.so add.o
上面兩步也可以直接從源文件一步到位:
gcc -fPIC -shared -o libadd.so add.c
編寫(xiě)測(cè)試代碼:
#include <stdio.h>
void main(){
printf("%d", add(1,2));
}
編譯成可執(zhí)行文件:
gcc test.c -o test -ladd -L ./
執(zhí)行發(fā)現(xiàn)報(bào)錯(cuò):
./test
error while loading shared libraries: libadd.so:
cannot open shared object file: No such file or directory
因?yàn)閳?zhí)行的時(shí)候找不到指定的動(dòng)態(tài)庫(kù)。
那我們把 libadd.so 放在執(zhí)行時(shí)的動(dòng)態(tài)庫(kù)默認(rèn)搜索路徑下,比如 /lib64:
cp libadd.so /lib64
再次執(zhí)行就成功了:
./test
3
查看二進(jìn)制文件的鏈接信息,也可以發(fā)現(xiàn)我們的 libadd.so 生效了:
ldd test
linux-vdso.so.1 => (0x00007ffe0f597000)
libadd.so => /lib64/libadd.so (0x00007fa5ab29f000)
libc.so.6 => /lib64/libc.so.6 (0x00007fa5aaed1000)
/lib64/ld-linux-x86-64.so.2 (0x00007fa5ab4a1000)
好了,以上就是編譯、靜態(tài)庫(kù)制作、動(dòng)態(tài)庫(kù)制作的過(guò)程,先記住這些 gcc 的基本常識(shí),再去研究原理吧!















 
 
 
 
 
 
 