【linux系统编程】编辑器gcc/g++

简介: 【linux系统编程】编辑器gcc/g++

Linux下的编辑器

介绍:

       Linux系统下可支持很多高级语言的编辑,不同的编辑工具对应不同的语言。本章,我们重点介绍C语言的编辑器 gcc 和C++的编辑器 g++,其中g++也可编辑C语言的文本,但gcc不可编辑C++的文本。


1,编辑器gcc/g++

       我们都知道,文件的编译过程需要经过预处理、编译、汇编、链接过程,直接 gcc [文件] 将会直接一次性执行完编辑文本的过程,形成可执行文件。

9cd4738d6f6b45b99c0902321647cee5.png



       上图中我们形成的可执行文件是系统默认命名的,使用 gcc code.c -o code.exe 或 gcc -o code.exe code.c 都将可执行文件命名为code.exe,即凡是跟在 -o 选项后面的名称都是可执行文件的命名。


4048bcf3c4e04d258e3eef15fa17e682.png


1-1,系统的编译过程

       众所周知,计算机的语言发展先是从二进制语言到汇编语言,然后不断发展,有了许多的高级语言,因此,高级语言在计算机内部执行时,需要将其翻译成汇编语言,然后从汇编语言翻译成二进制语言,因为发展过程都是要先有语言然后才有了编译器。


       当对文本进行编辑工作时,首先,在预处理过程中,系统会将头文件展开(本质是将头文件内容拷贝至源文件)、去掉文本中的注释、进行宏的替换、条件编译等。下一步进入编译工作时,系统会将高级语言编译成汇编语言,然后进入汇编阶段中,系统将汇编语言编译成二进制文件,此二进制文件称为 “可重定位目标二进制文件” ,注意,此时的二进制文件是不可执行的。因为此时的二进制文件还没有与高级语言中特定的库文件形成关联,即没有联系,而链接过程就是使用标准库文件,将形成的二进制文件与高级语言标准库文件中的代码形成关联,使系统能够识别其中代码的功能。


库文件:


       库文件是预先编译好的函数功能的集合,可以将用户写好的程序打包成一个整体,当其他模块或用户使用时,只需要这个库文件即可,不需要源代码文件。其中,库文件主要分为静态库和动态库,程序在使用库时使用静态链接或动态链接进行静态库或动态库的关联。Linux中静态库文件以 .a 后缀名,动态库文件以 .so 后缀名;Windows中静态库文件以 .lib 为后缀名,动态库文件以 .dll 为后缀名。


       静态库的特点是程序在编译连接的过程当中,编译器从库文件中取得所需代码,复制到生成的可执行文件中。因此静态库是程序在链接阶段被复制到程序当中,和程序的执行过程没有关系。


       动态库(也称为共享库)的特点是在程序运行时由系统动态加载到内存中,程序在使用时直接关联。因此,动态库可以被多个程序共享,同时只占用一份内存空间,提高了内存的使用效率。


       我们可使用 ldd 指令来观察可执行文件所依赖的库文件。


[zhu@zhujunhao ~]$ ldd code   #查看C语言中可执行文件code所依赖的库文件
    linux-vdso.so.1 =>  (0x00007ffd9b12a000)
    libc.so.6 => /lib64/libc.so.6 (0x00007fde6069b000)  #code依赖动态库/lib64/libc.so.6
    /lib64/ld-linux-x86-64.so.2 (0x00007fde60a69000)


       /lib64/libc.so.6这个文件就是C语言标准动态库,标准静态库是/lib64/libc.a,因此在Linux中,我们可直接使用C语言。其它语言也同理,平台要支持一种语言的开发,就必须要提前在系统中安装这种语言的标准头文件和库文件。


1-2,预处理过程

       当我们只想要查看预处理生成的内容时,使用 gcc -E [目标文件] 可在直接显示预处理处理后生成的数据,使用 gcc -E [目标文件] -o [文件] (这里选项不分前后顺序)会将源文件处理后的数据放入 [文件] 中。当不存在 [文件] 时,系统会自动创建此文件。这里的 -E 表示系统将预编译过程处理完后停止。


[zhu@zhujunhao ~]$ gcc -E code.c -o code.i   #将 code.c 文件经预处理后生成的数据放入 code.i 中
[zhu@zhujunhao ~]$ ll
total 36
-rw-rw-r-- 1 zhu zhu    89 Dec  6 08:46 code.c
-rw-rw-r-- 1 zhu zhu 25606 Dec  6 08:47 code.i
-rw-rw-r-- 1 zhu zhu   827 Dec  5 14:52 install.sh


       在预处理过程,我们可使用 -D [宏名称] 自动将宏加入源文件中,在预处理阶段也会将其一并处理。这里要注意,这里的指令添加只是暂时将其添加,再次进入文本后可观察到没有此宏。


[zhu@zhujunhao ~]$ gcc code.c -DSETVECTOR=1 -o code.exe
[zhu@zhujunhao ~]$ ./code.exe
have SETVECTOR=1      #可发现存在此宏
预编译过程将添加的宏进行替换,但在其它阶段不可,因为只有在预编译过程中才处理宏
[zhu@zhujunhao ~]$ gcc -E code.c -D SETVECTOR -o code.i


       因此,通过以上可发现,我们在系统编程的过程中,我们可实现自己代码和编译器指令的动态裁剪。


1-3,编译过程

       当我们只想要观看编译处理后的内容时,需使用  -S 选项,使用方法跟预处理使用的语法和方法一样,即:gcc -S [目标文件] -o [文件](这里选项不分前后顺序),将处理完的数据放到 [文件] 中,不同的是,[目标文件] 可以是C语言源文件,也可以是预处理后的文件。-S 表示系统将编译过程处理完后就停止。


[zhu@zhujunhao ~]$ gcc -E code.c -o code.i     #预处理后的文件code.i
[zhu@zhujunhao ~]$ gcc -S code.c -o code.s    #处理C语言源文件,形成文件code.s
[zhu@zhujunhao ~]$ gcc -S code.i -o code.s1   #处理预处理的文件,形成文件code.s1


       此过程结束后将会形成汇编语言,即系统调用汇编编译器形成汇编语言。


1-4,汇编过程

       只观看汇编处理后的数据对应的选项为 -c ,用法一样,指令为 gcc -c [目标文件] -o [文件](这里选项不分前后顺序),将汇编形成的二进制数据放入 [文件] 中,其中 [目标文件] 可以是C语言源文件,可以是预处理后的文件,可以是编译过程中形成的汇编文件。-c 表示当系统将汇编过程处理完后就停止。


[zhu@zhujunhao ~]$ gcc -c code.c -o code.o  #处理C语言源文件code.c
[zhu@zhujunhao ~]$ gcc -c code.i -o code.o   #处理预处理后的文件code.i
[zhu@zhujunhao ~]$ gcc -c code.s -o code.o  #处理编译后的汇编文件code.s

       上面的形成的 code.o 文件就是二进制文件。


1-5,链接过程

       链接是系统处理的最后一步,这里不需要任何选项,直接 gcc 编译即可,即 gcc [目标文件] -o [文件] (这里选项不分前后顺序),这里形成的文件就是将二进制文件与标准库文件链接后的可执行文件。这里表示的是系统直接处理 [目标文件] ,形成可执行文件。


[zhu@zhujunhao ~]$ gcc code.o -o code    #将二进制文件code.o链接


       可发现,这就是我们平常直接将源文件形成可执行文件的指令。即链接指令。


       链接过程中,系统默认使用动态链接,C语言的动态库在 /lib64/libc.so.6 文件下,而C++的动态库在  /lib64/libstdc++.so.6 文件下,其中,使用 -static 可在链接时改为静态链接,这里要注意的是有些系统默认可能没有安装静态库。我们需要去安装静态库,安装C和C++的静态库指令如下:


[root@zhujunhao ~]# yum install -y glibc-static libstdc++-static   
glibc-static 是C语言的静态库         libstdc++-static是C++的静态库

       其中,C语言安装后的静态库在 /lib64/libc.a 文件下,C++的静态库在 /lib64/libstdc++.a 文件下。


       链接过程使用静态链接指令如下:


[zhu@zhujunhao ~]$ gcc code.c -o code-static -static
[zhu@zhujunhao ~]$ ldd code-static
    not a dynamic executable    #查看后可发现链接的不是动态库,即链接静态库


       g++的使用都与gcc一样,注意的是系统原本没有安装 g++ 编译器,在 CentOS 版本下,安装g++ 相关的指令是:yum install gcc-c++


相关文章
|
7月前
|
Ubuntu Linux Anolis
Linux系统禁用swap
本文介绍了在新版本Linux系统(如Ubuntu 20.04+、CentOS Stream、openEuler等)中禁用swap的两种方法。传统通过注释/etc/fstab中swap行的方式已失效,现需使用systemd管理swap.target服务或在/etc/fstab中添加noauto参数实现禁用。方法1通过屏蔽swap.target适用于新版系统,方法2通过修改fstab挂载选项更通用,兼容所有系统。
650 3
Linux系统禁用swap
|
7月前
|
Linux
Linux系统修改网卡名为eth0、eth1
在Linux系统中,可通过修改GRUB配置和创建Udev规则或使用systemd链接文件,将网卡名改为`eth0`、`eth1`等传统命名方式,适用于多种发行版并支持多网卡配置。
1202 3
|
Ubuntu Linux 网络安全
Linux系统初始化脚本
一款支持Rocky、CentOS、Ubuntu、Debian、openEuler等主流Linux发行版的系统初始化Shell脚本,涵盖网络配置、主机名设置、镜像源更换、安全加固等多项功能,适配单/双网卡环境,支持UEFI引导,提供多版本下载与持续更新。
749 3
Linux系统初始化脚本
|
7月前
|
安全 Linux Shell
Linux系统提权方式全面总结:从基础到高级攻防技术
本文全面总结Linux系统提权技术,涵盖权限体系、配置错误、漏洞利用、密码攻击等方法,帮助安全研究人员掌握攻防技术,提升系统防护能力。
897 1
|
7月前
|
监控 安全 Linux
Linux系统提权之计划任务(Cron Jobs)提权
在Linux系统中,计划任务(Cron Jobs)常用于定时执行脚本或命令。若配置不当,攻击者可利用其提权至root权限。常见漏洞包括可写的Cron脚本、目录、通配符注入及PATH变量劫持。攻击者通过修改脚本、创建恶意任务或注入命令实现提权。系统管理员应遵循最小权限原则、使用绝对路径、避免通配符、设置安全PATH并定期审计,以防范此类攻击。
1269 1
|
7月前
|
Linux 应用服务中间件 Shell
二、Linux文本处理与文件操作核心命令
熟悉了Linux的基本“行走”后,就该拿起真正的“工具”干活了。用grep这个“放大镜”在文件里搜索内容,用find这个“探测器”在系统中寻找文件,再用tar把东西打包带走。最关键的是要学会使用管道符|,它像一条流水线,能把这些命令串联起来,让简单工具组合出强大的功能,比如 ps -ef | grep 'nginx' 就能快速找出nginx进程。
834 1
二、Linux文本处理与文件操作核心命令
|
7月前
|
Linux
linux命令—stat
`stat` 是 Linux 系统中用于查看文件或文件系统详细状态信息的命令。相比 `ls -l`,它提供更全面的信息,包括文件大小、权限、所有者、时间戳(最后访问、修改、状态变更时间)、inode 号、设备信息等。其常用选项包括 `-f` 查看文件系统状态、`-t` 以简洁格式输出、`-L` 跟踪符号链接,以及 `-c` 或 `--format` 自定义输出格式。通过这些选项,用户可以灵活获取所需信息,适用于系统调试、权限检查、磁盘管理等场景。
482 137
|
7月前
|
安全 Ubuntu Unix
一、初识 Linux 与基本命令
玩转Linux命令行,就像探索一座新城市。首先要熟悉它的“地图”,也就是/根目录下/etc(放配置)、/home(住家)这些核心区域。然后掌握几个“生存口令”:用ls看周围,cd去别处,mkdir建新房,cp/mv搬东西,再用cat或tail看文件内容。最后,别忘了随时按Tab键,它能帮你自动补全命令和路径,是提高效率的第一神器。
1296 58
|
6月前
|
存储 安全 Linux
Linux卡在emergency mode怎么办?xfs_repair 命令轻松解决
Linux虚拟机遇紧急模式?别慌!多因磁盘挂载失败。本文教你通过日志定位问题,用`xfs_repair`等工具修复文件系统,三步快速恢复。掌握查日志、修磁盘、验重启,轻松应对紧急模式,保障系统稳定运行。
1185 2