C程序编程四步走(c编程技巧)

任何一个 C 程序代码到生成一个可执行文件都需要四步,分别是预处理 Pre-processing ,编译 Compiling ,汇编 Assembling 和链接 Link ,这里借助 Gcc 工具来探究这四步分别做了什么事,起到什么样的作用。本文使用的测试代码是经典入门程序 "Hello World!"。

测试环境

为探究预处理,编译,汇编和链接的功能,我们在 Ubuntu 系统中使用 gcc 编译器( version=4.8.4 ),简单的也是最经典的入门程序 "Hello World!" 作为测试代码。源文件 hello.c[1] 代码如下:

// filename: hello.c# include <stdio.h>int main(void){ printf("Hello World!"); return 0;}

正常情况我们都会执行命令 gcc hello.c -o hello.out 来生成二进制可执行程序 hello.out。

预处理[2]

C 预处理器是用在编译器处理程序之前,它预扫描源代码完成包含头文件宏扩展条件编译行控制等功能。对于测试代码中,预处理器只对头文件进行了处理。获取预处理器输出的结果使用该命令 gcc -E hello.c -o hello.i。由于 hello.i[3] 文件内容比较多,这里截取部分进行说明。

// filename: hello.i# 1 "hello.c"# 1 "<built-in>"# 1 "<command-line>"# 1 "/usr/include/stdc-predef.h" 1 3 4# 1 "<command-line>" 2# 1 "hello.c"...# 1 "/usr/lib/gcc/x86_64-linux-gnu/4.8/include/stddef.h" 1 3 4# 212 "/usr/lib/gcc/x86_64-linux-gnu/4.8/include/stddef.h" 3 4typedef long unsigned int size_t;...# 5 "hello.c" 2int main(){ printf("Hello World!"); return 0;}

Tips:
hello.i 中有很多这样的格式
# line filename flags,它表示下面行是由文件 filename 的第 line 行生成的。其中 flags 有 1,2,3,4 四种取值

• 1 代表新文件的开始

• 2 代表返回一个文件

• 3 代表下面的文本来自系统头文件,所以某些警告可以过滤掉

• 4 代表下面的文本应该包含在extern C块中 按照提示 stddef.h 文件中第 212 行有 size_t 的宏定义。

编译[4]

编译的过程是将某种编程语言写的源代码(这里特指 C 语言)转换成另一种编程语言(这里特指汇编语言)。前面我们将 hello.c 预处理成了 hello.i 文件,现在就要将 hello.i 文件编译成汇编文件 hello.s 。获取编译器输出的结果使用命令 gcc -S hello.i -o hello.s 。汇编结果见 hello.s[5]

.file "hello.c" .section .rodata .LC0: .string "Hello World!" .text .globl main .type main, @function main: .LFB0: .cfi_startproc pushq %rbp .cfi_def_cfa_offset 16 .cfi_offset 6, -16 movq %rsp, %rbp .cfi_def_cfa_register 6 movl $.LC0, �i movl $0, �x call printf movl $0, �x popq %rbp .cfi_def_cfa 7, 8 ret .cfi_endproc .LFE0: .size main, .-main .ident "GCC: (Ubuntu 4.8.4-2ubuntu1~14.04.3) 4.8.4" .section .note.GNU-stack,"",@progbits

汇编[6]

汇编的过程是将汇编语言编写的源码转换成可执行的机器代码,通常目标文件中包含至少两个段:代码段和数据段。其中代码段包含程序的指令,一般可读和可执行,不可写;数据段用来存放程序中所用到的各种全局变量或静态数据,一般可读,可写,可执行。获取汇编器输出的结果使用该命令 gcc -o hello.o -c hello.c ,由于 hello.o 是二进制文件,是无法阅读的。这里我们通过命令 objdump 来对二进制文件进行反汇编,查看里面内容。

// objdump -d hello.o 查看hello.o中代码段信息hello.o:文件格式 elf64-x86-64Disassembly of section .text:0000000000000000 <main>: 0: 55 push %rbp 1: 48 89 e5 mov %rsp,%rbp 4: bf 00 00 00 00 mov $0x0,�i 9: b8 00 00 00 00 mov $0x0,�x e: e8 00 00 00 00 callq 13 <main 0x13> 13: b8 00 00 00 00 mov $0x0,�x 18: 5d pop %rbp 19: c3 retq

hello.o中各段信息如下:

// objdump -h hello.o 显示hello.o中各个段的头部信息hello.o:文件格式 elf64-x86-64节:Idx Name Size VMA LMA File off Algn 0 .text 0000001a 0000000000000000 0000000000000000 00000040 2**0 CONTENTS, ALLOC, LOAD, RELOC, READONLY, CODE 1 .data 00000000 0000000000000000 0000000000000000 0000005a 2**0 CONTENTS, ALLOC, LOAD, DATA 2 .bss 00000000 0000000000000000 0000000000000000 0000005a 2**0 ALLOC 3 .rodata 0000000d 0000000000000000 0000000000000000 0000005a 2**0 CONTENTS, ALLOC, LOAD, READONLY, DATA 4 .comment 0000002c 0000000000000000 0000000000000000 00000067 2**0 CONTENTS, READONLY 5 .note.GNU-stack 00000000 0000000000000000 0000000000000000 00000093 2**0 CONTENTS, READONLY 6 .eh_frame 00000038 0000000000000000 0000000000000000 00000098 2**3 CONTENTS, ALLOC, LOAD, RELOC, READONLY, DATA

链接[7]

链接的过程是将一个或多个由编译器或汇编器生成的目标文件链接库(静态库或动态库)形成可执行文件。其中静态库会和汇编生成的目标文件一起链接打包到可执行文件中(静态链接),它对函数库的链接是放在编译时期完成的。而动态库在程序编译时不会被链接到可执行文件中,而是在程序运行时才会被载入(动态链接)。不同的应用程序如果调用相同的库,那么在内存中只需要一份该共享库实例。获取链接器链接后的可执行文件使用命令 gcc hello.o -o hello 。如果想看该可执行文件依赖的库,可以使用命令 ldd hello

# ldd hello 显示hello依赖的库 linux-vdso.so.1 => (0x00007ffc85980000) libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f06c7a53000) /lib64/ld-linux-x86-64.so.2 (0x000055ad7be9e000)

参考文献

1.预处理[8]

2.预处理-行号标记[9]

3.编译器[10]

4.汇编[11]

5.链接器[12]

References

[1] hello.c: https://blog.haojunyu.com/atts/hello.c
[2] 预处理:
https://zh.wikipedia.org/wiki/C预处理器
[3] hello.i:
https://blog.haojunyu.com/atts/hello.i
[4] 编译:
https://zh.wikipedia.org/wiki/編譯器
[5] hello.s:
https://blog.haojunyu.com/atts/hello.s
[6] 汇编:
https://zh.wikipedia.org/wiki/汇编语言
[7] 链接:
https://zh.wikipedia.org/wiki/链接器
[8] 预处理:
https://zh.wikipedia.org/wiki/C预处理器
[9] 预处理-行号标记:
https://gcc.gnu.org/onlinedocs/cpp/Preprocessor-Output.html
[10] 编译器:
https://zh.wikipedia.org/wiki/編譯器
[11] 汇编:
https://zh.wikipedia.org/wiki/汇编语言
[12] 链接器:
https://zh.wikipedia.org/wiki/链接器

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

(0)
上一篇 2023年5月7日 上午8:27
下一篇 2023年5月7日 上午8:37

相关推荐

  • 2021年 信息系统项目管理师

    2021年信息系统项目管理师考试将于近日举行,这一考试是对信息系统项目管理师的认证考试,也是全球信息系统领域中最重要的考试之一。作为专业的信息系统项目管理师,需要具备深厚的技术背景…

    科研百科 2024年7月24日
    59
  • 建筑工程安全管理制度的分享(建筑安全制度与管理)

    第一章 安全生产教育培训制度 一、安全目标 我国安全生产方针:安全第一、预防为主、综合治理的原则。 ①在作业过程中,应当严格遵守本单位的安全生产规章制度和操作规程,服从管理,正确佩…

    科研百科 2022年12月6日
    236
  • 各区动态 – 浦东新区召开基层应急管理能力标准化建设示范点观摩会

    为深入贯彻落实党的二十大精神和习近平总书记关于应急管理工作的重要论述,加快推进基层应急管理体系和能力现代化,加强基层应急管理工作的规范性、科学性和有效性,11月15日下午,浦东新区…

    科研百科 2023年1月30日
    255
  • 清风头条丨“被遗忘的”生态公益林资金?(生态公益林保护)

    红网时刻通讯员 赵贤博 记者 王义正 湘潭报道 “焦书记,你们纪委能帮我查查我的生态公益林补贴去哪儿了吗?”近日,湘潭市湘潭县石鼓镇纪委书记焦乐平的办公室接待了一名贫困户。 原来,…

    2022年6月30日
    344
  • 中交协同办公系统

    中交协同办公系统: 提升中交企业效率的利器 随着中交企业的不断壮大和发展,协同办公系统已经成为中交企业必不可少的工具之一。中交协同办公系统可以帮助中交企业更好地管理和协调各部门的工…

    科研百科 2024年8月22日
    44
  • VC|MFC的CDC类及如何使用画笔绘制各种图形(vc应用mfc画图的详细步骤)

    CDC对象提供处理显示器或打印机等设备上下文的成员函数,以及处理与窗口客户区对应的显示上下文的成员。另外还为绘制文本、处理字体,使用打印机跳转,滚动和播放元文件提供成员函数。 它还…

    科研百科 2024年4月29日
    80
  • 项目管理 深圳

    项目管理深圳:城市创新与发展的挑战与机遇 随着中国经济的快速发展,深圳成为了中国最具活力的城市之一。作为经济特区和全国性经济中心城市,深圳在项目管理方面面临着许多挑战和机遇。本文将…

    科研百科 2024年8月16日
    33
  • 全面加强党的组织体系建设(全面加强党的组织体系建设心得体会)

    加强党的组织体系建设,是习近平总书记从战略和全局高度提出的重大命题。我们要认真领会,结合实际努力在工作中破题答题,创造性地贯彻落实。 深刻理解党的组织体系的科学内涵 党的组织体系是…

    科研百科 2023年7月9日
    191
  • 工程项目三大项目管理

    工程项目三大项目管理 项目管理在工程项目中起着至关重要的作用。它包括规划、执行和控制三个主要阶段。这三个阶段是相互关联的,并且必须协调一致,以确保项目能够按时、按预算和按规格完成。…

    科研百科 2024年8月27日
    30
  • 绿城项目管理

    绿城项目管理: 一个追求卓越的项目管理实践 绿城项目管理是中国项目管理领域的知名企业之一,其项目管理实践在业界享有很高的声誉。本文将介绍绿城项目管理在项目管理方面的实践和成就。 绿…

    科研百科 2024年7月27日
    52