执行详解:
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
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 <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"在单步执行的时候,只有当前线程会执行;
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;
}
注意:顶格与tab键
了解makefile文件的生成规则;
总结:
makefile可以提高编译效率,使用make命令每次都只会编译那些修改了的或者依赖修改了的文件(间接修改),没有修改的文件不会重新编译;