当前位置: 首页 > 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值改变

相关文章:

  • Linux账号和权限管理
  • U盘挂载Linux
  • 道可云人工智能每日资讯|北京农业人工智能与机器人研究院揭牌
  • LabelMe安装踩坑
  • 电子电路:什么是晶振?
  • python第31天打卡
  • [野火®]《FreeRTOS 内核实现与应用开发实战—基于STM32》笔记
  • cf1600-1900每天刷2-3道打卡(2)
  • 黑盒(功能)测试基本方法
  • LARWINER拉威兒艺术珠宝携手郭培GUOPEI高定服装 共谱「宝光凝粹,锦绣华裳」
  • OpenCV CUDA模块图像处理------图像融合函数blendLinear()
  • ChatGPT实战嵌入式开发应用指南与代码演示
  • 2025中国主流大模型全景解析:技术路线、场景实践与生态博弈
  • docker中启动 Python 程序并调用某个模块内的函数的方法
  • centos安装locate(快速查找linux文件)
  • SMART原则讲解
  • docker-compose 方式搭建 Jpom
  • 关于线缆行业设备数据采集异构问题的解决
  • 深度思考:摆脱无效忙碌的核心策略
  • 【Typst】2.Typst标记语法和基础样式
  • 河北网站建设及推广/搜狗引擎搜索
  • wordpress个人博客建站/seo管家
  • 盘锦市建设银行网站/平台推广方式
  • 兰溪市住房和城乡建设局网站/搜索引擎优化自然排名
  • 中企动力做网站/青岛网站开发公司
  • 查企业法人信息查询平台/seo优化工具哪个好