Linux的gcc,gdb基础

执行详解:

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
#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

gdb调试

(1)debug版本:


编译阶段会加入某些调试信息;
调试信息是在编译的过程中加入到中间文件.o文件的;
gcc -c main.c -g:生成包含调试信息的中间文件
gcc -o main main.o
一步执行:gcc -o main main.c -g


(2)release版本:


发行版本,没有调试信息;
gcc默认生成release版本;


(3)gdb基础命令:


gdb 可执行文件名

显示代码: l

加断点: b 行号
启动程序:r(运行之前一定要加断点)
查看断点信息: info break/info b
删除断点信息:delete 断点编号
单步执行:n
打印 :p
显示:display 变量名:
退出:q

示例1:

#include
 #include

 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"在单步执行的时候,只有当前线程会执行;

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
#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;
}

Linux的gcc,gdb基础_第1张图片

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

你可能感兴趣的:(Linux,linux,运维,服务器)