深圳幻海软件技术有限公司 欢迎您!

Java 程序员眼里的 Gcc

2023-02-28

作为一名Java程序员,对gcc的基本使用总是记不住,很不熟练,今天写篇文章用最简单的方式记录下。编译的过程写个hello.c代码:复制#include<stdio.h>#defineSTR"helloworld\n"voidmain(){printf(STR);}1.2.3.4.5.6

作为一名 Java 程序员,对 gcc 的基本使用总是记不住,很不熟练,今天写篇文章用最简单的方式记录下。

编译的过程

写个 hello.c 代码:

#include <stdio.h>
#define STR "hello world\n"

void main() {
    printf(STR);
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.

第一步:预处理(preprocess)

其实就是以下三个动作:

  • 头文件展开
  • 宏定义直接替换
  • 条件编译不成立就去掉
gcc -E hello.c -o hello.i
  • 1.

第二步:编译(compile)

转换成汇编语言:

gcc -S hello.i -o hello.s
  • 1.

第三步:汇编(assemble)

转换成二进制(ELF relocatable):

gcc -c hello.s -o hello.o
  • 1.

第四步:链接(link)

具体分为动态链接和静态链接:

# 动态链接
gcc hello.o -o hello
# 静态链接
gcc hello.o -o hello -static
  • 1.
  • 2.
  • 3.
  • 4.

静态库制作

写一个加法函数:

int add(int a, int b) {
    return a+b;
}
  • 1.
  • 2.
  • 3.

编译成 .o:

gcc -c add.c -o add.o
  • 1.

制作成静态库:

ar rcs libadd.a add.o
  • 1.

编写测试代码:

#include <stdio.h>
void main(){
    printf("%d", add(1,2));
}
  • 1.
  • 2.
  • 3.
  • 4.

编译成可执行文件:

#写法一
gcc test.c -o test libadd.a
#写法二
gcc test.c -o test -static -ladd -L ./
  • 1.
  • 2.
  • 3.
  • 4.

执行:

./test
3
  • 1.
  • 2.

动态库制作

写一个加法函数:

int add(int a, int b) {
    return a+b;
}
  • 1.
  • 2.
  • 3.

编译成 .o:

gcc -c add.c -o add.o -fPIC
  • 1.

制作成动态库:

gcc -shared -o libadd.so add.o
  • 1.

上面两步也可以直接从源文件一步到位:

gcc -fPIC -shared -o libadd.so add.c
  • 1.

编写测试代码:

#include <stdio.h>
void main(){
    printf("%d", add(1,2));
}
  • 1.
  • 2.
  • 3.
  • 4.

编译成可执行文件:

gcc test.c -o test -ladd -L ./
  • 1.

执行发现报错:

./test
error while loading shared libraries: libadd.so: 
cannot open shared object file: No such file or directory
  • 1.
  • 2.
  • 3.

因为执行的时候找不到指定的动态库。

那我们把 libadd.so 放在执行时的动态库默认搜索路径下,比如 /lib64:

cp libadd.so /lib64
  • 1.

再次执行就成功了:

./test
3
  • 1.
  • 2.

查看二进制文件的链接信息,也可以发现我们的 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)
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.

好了,以上就是编译、静态库制作、动态库制作的过程,先记住这些 gcc 的基本常识,再去研究原理吧!