【static&extern】用GCC输出带源代码的汇编程序进行链接的调试

本文通过一个简单的示例展示了如何定位并解决C/C++程序中由于符号未定义导致的链接错误。通过对编译过程的深入剖析,揭示了使用静态限定符可能导致的问题,并提供了具体的调试方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

引用请注明出处:http://blog.youkuaiyun.com/int64ago/article/details/7395418

一个可执行程序的形成,大致的步骤:编译——>链接——>可执行文件,然而,犯错最多的地方可能你以为是编译阶段,但是,实际上是链接阶段,特别是大型工程。下面用一个例子简单说下出现链接错误的一个简单解决方式,先看两个程序:

  1. //main.c  
  2. void test1();  
  3. void test2();  
  4.   
  5. int main()  
  6. {  
  7.     test1();  
  8.     test2();  
  9.     return 0;  
  10. }  

  1. //test.c  
  2. #include <stdio.h>  
  3.   
  4. static void test1()  
  5. {  
  6.     printf("This is test1\n");  
  7. }  
  8.   
  9. void test2()  
  10. {  
  11.     printf("This is test2\n");  
  12.     test1();  
  13. }  

这两个简单的程序也没干啥,就是用来测试的,先编译:

gcc -c main.c

gcc -c test.c

没有错误,生成了main.o 和 test.o两个目标文件,接下来链接:

gcc -o main main.o test.o

出现了下面的错误:

main.o: In function `main':
main.c:(.text+0x7): undefined reference to `test1'
collect2: ld returned 1 exit status

意思是没有定义test1,可能此时你已经知道了错误的原因及地方,可以改了,但是试想一下如果工程巨大,很多地方用了test1,你这时就需要另外方法了(除非你用的是IDE,这里我们不讨论)。再仔细看看,还给了另一条信息:main.c:(.text+0x7),这就是出错位置,但是这时汇编的位置,所以我们要深入汇编,可以用下面的命令:

gcc -c -g -Wa,-adlhn main.c >/dev/null

objdump -S main.o >log

查看log:

  1. main.o:     file format elf32-i386  
  2.   
  3.   
  4. Disassembly of section .text:  
  5.   
  6. 00000000 <main>:  
  7.   
  8. void test1();  
  9. void test2();  
  10.   
  11. int main()  
  12. {  
  13.    0:   55                      push   %ebp  
  14.    1:   89 e5                   mov    %esp,%ebp  
  15.    3:   83 e4 f0                and    $0xfffffff0,%esp  
  16.     test1();  
  17.    6:   e8 fc ff ff ff          call   7 <main+0x7>  
  18.     test2();  
  19.    b:   e8 fc ff ff ff          call   c <main+0xc>  
  20.     return 0;  
  21.   10:   b8 00 00 00 00          mov    $0x0,%eax  
  22. }  
  23.   15:   c9                      leave    
  24.   16:   c3                      ret      
就会发现.text+0x7的位置是在调用test1()的地方,现在到test.c会发现test1()的类型前加了static,这就显示的告诉编译器这个函数只在本文件可见,这里当然还涉及到的static、extern等类型修饰字有空再开贴说明……
### 使用 GCC 链接静态库 当使用 `gcc` 编译程序并链接静态库时,需要指定编译器查找静态库的位置以及具体的库文件名。对于静态库而言,在命令行中通常会通过 `-L` 参数来指明库所在的路径,并利用 `-l` 来声明要链接的具体库名称。 例如,如果想要链接数学库(假设位于 `/usr/lib/` 下),可以在编译命令里加入如下选项: ```bash gcc myprogram.c -o myprogram -L/usr/lib -lm ``` 这里 `-L/usr/lib` 表示告诉编译器去该目录下寻找所需的 `.a` 文件;而 `-lm` 则是指定链接名为 `m` 的静态库,即 `libm.a`[^2]。 另外,值得注意的是,静态库的名字一般遵循特定模式——前缀为 "lib", 后缀为 ".a" 或者其他平台特有的扩展名。因此在实际操作过程中只需给出中间部分作为参数传递给 `-l` 即可。 为了更清晰地展示整个过程,下面提供了一个完整的例子,其中包含了创建源码文件、编译目标文件以及最终链接成可执行文件的过程: #### 创建 C 源代码文件 (`main.c`) ```c #include <stdio.h> extern void hello_world(); int main() { printf("Before calling function from static lib\n"); hello_world(); return 0; } ``` #### 创建用于构建静态库的C源代码(`hello_func.c`) 和头文件(`hello_func.h`) ##### hello_func.c ```c #include "hello_func.h" void hello_world(){ printf("Hello world! This message comes from a static library.\n"); } ``` ##### hello_func.h ```c #ifndef HELLO_FUNC_H_ #define HELLO_FUNC_H_ void hello_world(); #endif /*HELLO_FUNC_H_*/ ``` #### 构建静态库 先将上述功能函数编译为目标文件: ```bash gcc -c hello_func.c -o hello_func.o ``` 接着把目标文件打包进一个档案(.a),形成静态库: ```bash ar rcs libhello.a hello_func.o ``` #### 将应用程序与自定义静态库一起编译 最后一步就是编写主程序并与刚制作好的静态库相连接: ```bash gcc main.c -L. -lhello -o app_with_static_lib ``` 此命令中的 `-L.` 告诉编译器当前工作目录也应被搜索以找到任何提及到的`.a` 形式的库;`-lhello` 是用来指示链接名为 `hello` 的静态库 (全称为 `libhello.a`)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值