Linux 中gcc与gdb

1.执行详解
1)如何执行
路径+可执行文件名 或者 路径+可执行文件名 & (将进程放到后台执行);
可以把可执行文件放到 /usr/bin 就可以省略路径了;
思考:为什么?
ps :/usr/bin ps,ls,pwd  (先了解,后期写项目就知道为什么了)


2)两步执行与一步执行
a.可以三步合为一步,即不经过预编译,编译,汇编三步,直接一步生成.o文件: gcc -c main.c -o main.o gcc -o main main.o

gcc -c main.c(只编译,看有没有语法错误);工作中常用
b.可以四步合为一步: gcc -o main main.c


3)多文件的编译执行
先写如下三个文件: add.c max.c main.c

//add.h
int add(int x,int y);

//add.c
int add(int x,int y)
{
    return x+y;
} 

//max.h
int max(int x,int y);

//max.c
int max(int x,int y)
{
     return x>y?x:y;
} 


//main.c
#include <stdio.h>
#include "./max.h"
#include "./add.h"
int main()
{
    int a=10;
    int b=20;
    printf("a+b=%d\n",add(10,20));
    printf("a,b的最大值为%d\n",max(10,20));
    return 0;
}

如何执行呢?
两步执行:
gcc  -c  main.c
gcc  -o main main.c  add.c  max.c
gcc  -o main main.c (error)
一步执行:gcc  -o main main.c  add.c  max.c
2.gdb调试
(1)debug版本: 在编译阶段会加入某些调试信息; 调试信息是在编译的过程中加入到中间文件.o文件的; gcc -c main.c -g:生成包含调试信息的中间文件 gcc -o main main.o 一步执行:gcc -o main main.c -g (1)

(2)release版本: 发行版本,没有调试信息; gcc默认生成release版本; (3)gdb基础命令: gdb 可执行文件名 (2)

显示代码: l (3)
加断点: b 行号

(4) 启动程序:r(运行之前一定要加断点)

(5) 查看断点信息: info break/info b 删除断点信息:delete 断点编号 单步执行:n 打印 :p 显示:display 变量名:

(6) 退出:q

示例1:

#include <stdio.h>
 #include <string.h>

 int main()
{
    while(1)
   {
       char buff[128]={0};
       printf("input\n");
       fgets(buff,128,stdin);

       if(strcmp(buff,"end")==0)
       {
          break;
       }
       printf("read:%s",buff);
   }

    return 0;
 }

gdb命令(全):

l:显示main函数所在的文件的源代码

list 文件名:num 显示文件名文件num行上下的源代码(多文件)

b 行号:给指定行添加断点

b 函数名:给指定函数的第一有效行添加一个断点

info break:显示断点信息;

(info b) delete 断点号:删除指定断点

r(run):运行程序 n(next):单步执行

c(continue):继续执行,直接执行到下一个断点处

s:进入将要被调用的函数中执行 finish:跳出函数;

q:退出调试

bt:显示函数调用栈 disable 断点号:将断点设定为无效的,不加断点号,将所有断点设置为无效 enable 断点号:将断点设定为有效的,不加断点号,将所有断点设置为有效;

p val:打印变量val的值

p &val:打印变量val的地址

p a+b:打印表达式的值

p arr(数组名):打印数组所有元素的值

p   *arr@len:用指向数组的指针打印数组所有元素的值

display:自动显示,参数和p命令一样;

info display:显示自动显示信息

undisplay+编号:删除指定的自动显示

ptype val:显示变量类型

示例2:

int  SUM(int n)
{
    int sum=0;
    for(int i=0;i<=n;i++)
    {
        sum+=i;
    }
    return sum;
}

int main()
{
    int sum=SUM(100);
    printf("%d\n",sum);
    return 0;
}

补充命令: 多进程的调试命令:
(gdb) set follow-fork-mode mode mode可以选择parent或者child,即:选择调试哪个进程 注意:未被跟踪调试的进程会直接执行结束;
多线程调试命令:

1)利用info threads查看线程信息;

2)thread id:调试目标id指定的线程;

3)set scheduler-locking off|on|step;

"off"表示不锁定任何线程;

"on"只有当前被调试的线程继续运行;

"step"在单步执行的时候,只有当前线程会执行;

3.makefile安装及make
makefile文件:Linux上的工程管理工具,可以实现自动化编译; 工程中的源文件不计其数,可以根据模块,功能等存储在不同的目录中; makefile可以提高编译效率,使用make命令每次只会编译那些修改了的或者依赖修改了的这些文件,没有修改的文件不会重新编译. VS底层就有自己的makefile文件;

//add.h
int add(int x,int y);

//add.c
int add(int x,int y)
{
    return x+y;
} 

//max.h
int max(int x,int y);

//max.c
int max(int x,int y)
{
     return x>y?x:y;
} 


//main.c
#include <stdio.h>
#include "./max.h"
#include "./add.h"
int main()
{
    int a=10;
    int b=20;
    printf("a+b=%d\n",add(10,20));
    printf("a,b的最大值为%d\n",max(10,20));
    return 0;
}


 

image-20230712174853467.png

注意:顶格与tab键 了解makefile文件的生成规则; 总结: makefile可以提高编译效率,使用make命令每次都只会编译那些修改了的或者依赖修改了的文件(间接修改),没有修改的文件不会重新编译;


注意:顶格与tab键 了解makefile文件的生成规则; 总结: makefile可以提高编译效率,使用make命令每次都只会编译那些修改了的或者依赖修改了的文件(间接修改),没有修改的文件不会重新编译;

  • 34
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值