1.gcc编译
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;
}
自己动手写makefile文件,理解makefile:

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

被折叠的 条评论
为什么被折叠?



