当前位置: 首页 > news >正文

Linux-GCC、makefile、GDB

GCC

gcc -E test.c -o test.i预处理(-o指定文件名) 
gcc -S test.i -o test.s编译
gcc -c test.s -o test.o汇编
gcc test.o -o test链接(生成一个可执行程序的软连接)
gcc test.c -o test一条指令可以完成以上所有内容
gcc *.c -I(大写的i) include由于在main.c中找不到当前文件夹中的头文件,所以用-I指定在include文件夹中找对应的头文件。

head.h包含所有运算的函数声明。

以下代码由于程序中没有DEBUG宏,所以不会输出"我被执行"。

#include<stdio.h>int main(){#ifdef DEBUGprintf("我被执行\n");#endifprintf("hello world");return 0;
}
gcc test.c -D DEBUG由于局外定义了宏所以会输出"我被执行"。

制作静态库:

gcc -c add.c sub.c div.c mult.c -I(大写的i) ./include生成add.o sub.o div.o mult.o
ar rcs libcal.a *.o将所有.o文件打包(如果静态库要发布出去要有libcal.a 和 head.h两个文件)。

使用静态库:
head.h和libcal.a和main.c在同一个目录。

gcc main.c -o cal -L ./ -l(小写的L) cal-L指定库路径 -l指定库名称,掐头(lib)去尾(.a)
./cal执行

制作动态库:

gcc -c -fpic add.c sub.c mult.c div.c -I(大写的i) ./include生成.o文件
gcc -shared *.o -o libcal.so生成动态库

使用动态库:
head.h和libcal.so和main.c要在同一个目录。

gcc main.c -L ./ -l(小写L) cal -o app生成app可执行文件
./app执行
ldd app可以查看app文件所需的动态库。

 当app文件和libcal.so不在同一目录,执行app文件会报错。
解决方案:
一、

sudo vim /etc/ld.so.conf
添加新路径:动态库所在的路径
sudo ldconfig


二、

sudo ln -s /xxx/xxx/libxxx.so /user/lib/libxxx.so

makefile

当目录下有makefile文件和一系列.c文件。

gcc *.c -o app生成一个app可执行文件。
make由于上述过程过于繁琐,用make可以自动化编译。(生成.o文件和一个可执行程序)
make clean删除.o文件和那个可执行文件。

makefile的编写:
 

vim makefile
cal:add.c div.c main.c mult.c sub.cgcc add.c div.c main.c mult.c sub.c -o cal
make此时会生成cal可执行文件。 
./cal执行cal程序


但是以上规则效率太低。

修改:

cal:add.o div.o main.o mult.o sub.ogcc add.o div.o main.o mult.o sub.o -o caladd.o:add.cgcc add.c -cdiv.o:div.cgcc div.c -cmult.o:mult.cgcc mult.c -cmain.o:main.cgcc main.c -csub.o:sub.cgcc sub.c -c

自动变量:

$<表示依赖项中第一个依赖文件的名称。
$@表示目标文件的名称,包含文件扩展名。
$^依赖项中,所有不重复的依赖文件,这些文件之间以空格分开。
# 这是一个规则的普通写法
cal:add.o div.o main.o mult.o sub.ogcc add.o div.o main.o mult.o sub.o -o cal# 这是一个规则,用了自动变量
cal:add.o div.o main.o mult.o sub.ogcc $^ -o $@

模式匹配:

# %是一个通配符,匹配的是文件名
%.o:%.cgcc $< -c

以下代码太冗余:

cal:add.o div.o main.o mult.o sub.ogcc add.o div.o main.o mult.o sub.o -o caladd.o:add.cgcc add.c -cdiv.o:div.cgcc div.c -cmult.o:mult.cgcc mult.c -cmain.o:main.cgcc main.c -csub.o:sub.cgcc sub.c -c

修改后:

target=cal
obj=add.o div.o main.o mult.o sub.o$(target):$(obj)gcc $(obj) -o $(target)%.o:%.cgcc -c $^

函数:
一、wildcard

$(wildcard *.c ./sub/*.c)返回值格式:a.c b.c c.c d.c e.c f.c ./sub/aa.c ./sub/bb.c

二、patsubst

src = a.cpp b.cpp c.cpp e.cpp接下来要把变量src中的所有文件名的后缀从.cpp替换为.o
obj = $(patsubst %.cpp, %.o, $(src)) obj 的值为: a.o b.o c.o e.o

以下代码还可以优化:

target=cal
obj=add.o div.o main.o mult.o sub.o$(target):$(obj)gcc $(obj) -o $(target)%.o:%.cgcc -c $^

优化后:

target=cal# 搜索磁盘源文件
src=$(wildcard *.c)# 后缀的替换
obj=$(patsubst %.c, %.o, $(src))$(target):$(obj)gcc $(obj) -o $(target)%.o:%.cgcc -c $^

只有当在make后面写clean时才会执行clean

target=cal# 搜索磁盘源文件
src=$(wildcard *.c)# 后缀的替换
obj=$(patsubst %.c, %.o, $(src))$(target):$(obj)gcc $(obj) -o $(target)%.o:%.cgcc -c $^clean:rm $(obj) $(target)
make clean删除了.o和目标可执行文件

由于当目录中有了clean文件后,再执行make clean会出错,所以要把clean声明为伪目标。

修改后:

target=cal# 搜索磁盘源文件
src=$(wildcard *.c)# 后缀的替换
obj=$(patsubst %.c, %.o, $(src))$(target):$(obj)gcc $(obj) -o $(target)%.o:%.cgcc -c $^.PHONY:clean
clean:rm $(obj) $(target)

现在即使当前目录有clean文件也不影响。

make clean 

由于mkdir没有管理员权限无法执行,默认往后的所有语句都无法执行

target=cal# 搜索磁盘源文件
src=$(wildcard *.c)# 后缀的替换
obj=$(patsubst %.c, %.o, $(src))$(target):$(obj)gcc $(obj) -o $(target)%.o:%.cgcc -c $^.PHONY:clean
clean:mkdir arm $(obj) $(target)

想要rm执行要在mkdir前面加个-

target=cal# 搜索磁盘源文件
src=$(wildcard *.c)# 后缀的替换
obj=$(patsubst %.c, %.o, $(src))$(target):$(obj)gcc $(obj) -o $(target)%.o:%.cgcc -c $^.PHONY:clean
clean:-mkdir arm $(obj) $(target)

GDB

gcc main.c -g -Wall -o0 -o app生成一个调试的可执行文件app
gdb app进入调试的命令
l(小写的L)查看代码(回车继续往下查看)
q退出
set args 1 2 3传入参数1 2 3(如果有函数参数)
show args查看参数
run运行程序
g++ -g *.cpp -o app生成调试的可执行文件
gdb app进入调试的命令
l(小写的L) insert.cpp:19查看insert.cpp第19行和上下相关内容
l(小写的L) insertSort在insert.cpp文件中找到insertSort函数
set list 20设置显示行数为20行
show list查看显示行数
l(小写的L) test.cpp:main查看到test.cpp里面的main函数
b 16在第16行设置断点
b if i==5用于例如在for循环里面当i==5时停止
b insert.cpp:16在insert.cpp中的第16行设置断点
i b查看已设置的断点
d 1删除第1个断点
d 1-2删除第1-2断点
d 3 5删除3 和 5断点
dis 4设置第4个断点无效
dis 6-7设置6-7断点无效
ena 4使第4断点生效
ena 6-7使6-7断点生效
p i查看变量i
p/d i以十进制查看变量i
p/c以字符型查看变量i
p/f以浮点数查看变量i
ptype i查看变量i的类型
ptype array[i]查看array[i]的类型
ptype array查看array的类型
display i自动跟踪变量i
display array[i]自动跟踪变量array[i]
i display查看所有的自动跟踪
next单步调试
undisplay 1取消跟踪编号为1的变量
disable display 3设置编号为3的变量自动跟踪为无效
ena display 3设置编号为3的变量自动跟踪为有效

单步调试:
run开始执行程序,一个函数执行的地方打了断点

step执行函数体内容
finish跳出函数体(里面不能有断点,不然跳不出来)
next不执行这个函数体
until跳出循环体(里面不能有断点,要使它失效或删除)
set var i=5将变量i的值设为5,可用于for循环中,在for循环打断点,将i值改变

http://www.lqws.cn/news/106255.html

相关文章:

  • 【软考】计算机系统构成及硬件基础知识
  • docker中组合这几个命令来排查 import 模块失败 的问题
  • Nginx实战
  • 各个主要目录的功能 / Linux 常见指令
  • 词语翻译的三步法与背后的语言学思维
  • 技巧小结:外部总线访问FPGA寄存器
  • 【25.06】fabric进行caliper测试加环境部署
  • 嵌入式系统:从技术原理到未来趋势(驱动程序篇)
  • 预览pdf(url格式和blob格式)
  • Python Day42
  • xmake的简易学习
  • 一、无参数的函数调用- RSP,EAX寄存器,全局变量,INT类型和MOV,INC,SHL指令
  • Python中os模块详解
  • Spring Boot 自动配置原理:从入门到精通
  • webstrom中git插件勾选提交部分文件时却出现提交全部问题怎么解决
  • UGUI Text/TextMeshPro字体组件
  • Activity
  • 6.3本日总结
  • agent mode 代理模式,整体要求,系统要求, 系统指令
  • ABP-Book Store Application中文讲解 - Part 7: Authors: Database Integration
  • 『uniapp』把接口的内容下载为txt本地保存 / 读取本地保存的txt文件内容(详细图文注释)
  • WPS word 已有多级列表序号
  • 免费批量文件重命名软件
  • AI健康小屋+微高压氧舱:科技如何重构我们的健康防线?
  • KITTI数据集(计算机视觉和自动驾驶领域)
  • mobilnet v4 部署笔记
  • go语言基础|slice入门
  • C语言学习—数据类型20250603
  • 2025.6.3总结
  • Jpom:Java开发者的一站式自动化运维平台详解