文章目录
一.项目自动化构建工具-make/Makefile
1.背景
2. 举例
- 第一步编写一个.c/.cpp文件
vim mycode.cpp
编写完后退出该文件。
- 第二步创建一个Makefile文件
vim Makefile
退出该文件
- 输入make进行编译
可以看见编写完一个Makefile后,输入make指令后,就会编译你指定要比编译的文件(例:mycode.cpp),生成了对应可执行程序(mycode)。
这样可以免去写编译指令,完成项目自动化构建。在需要大量编译时候可以节省很多时间,直接一个make命令就能完成了。
那么要删除可执行程序怎么办呢?
其中 .PHONY:clean 是伪目标,只输入make时候,不会执行.PHONY: 后面的指令,需要make加 .PHONY: 后面的指令才会执行该伪目标。
make clean
只输入make时候,不会编译也不会显示伪目标
那么Makefile是怎么实现这些的呢?
3. 原理
Makefile存在意义是为了构建项目的(要做一件事情),那么就需要依赖关系和依赖方法。例如:你问你爸要钱,那么你们之间的依赖关系是你们是父子,表明依赖方法是你爸要给你钱。
所以Makefile内部要写的是:依赖关系和依赖方法。
- 依赖关系
Makefile的mycode依赖mycode.cpp。mycode是可执行程序,必须由源文件mycode.cpp编译获得。
所以 mycode:mycode.cpp 是依赖关系。
依赖方法是: g++ mycode.cpp -o mycode
表明可执行程序是由源文件编译获得,提供依赖方法。
注意的是:在写完依赖方法回车到下一行,一个是按[Tab]键,不然会报错。
.PHONY被关键字修饰的对象是一个伪目标,该目标总是能被执行。
可见即便mycode不存在了,clean指令依然被执行,不会报错。
甚至.PHONY可以修饰mycode,使编译始终被执行。
.PHONY:mycode
make是如何工作的,在默认的方式下,也就是我们只输入make命令。那么,
如果不用修饰.PHONY修饰mycode,能不能重复执行编译呢?
可见不能一直重复编译,报错:
`mycode' is up to date.
意思使mycode已经使最新的了。
- 那么是什么工具阻止了重复的命令?又是怎么知道不需要再编译的呢?
是g++/gcc阻止该命令。g++/gcc通过比较源文件和可执行程序的Modify时间,如果源文件Modify时间比可执行程序时间新,就会编译,否则不会编译。
- 查看文件三个时间
stat mycode.cpp
stat mycode
其中Modif:文件内容被修改的时间
Chang:文件内容属性最新一次被修改时间
Access:文件最后访问时间
那么我们进入到源文件mycode.cpp中后再退出会看到Access和Chang有时候会变化,有时候不会变化,Access一变化那么Chang就会变化,而Modif不会变化。
(注意:退出vim时候,底行模式时候,只能输入q,因为w是保存,相当于更改了文件的内容)
Access显示不是文件最后访问时间吗为什么时变时不变呢?而Chang会随着Access变化而变化呢?
因为文件操作的时候,文件的访问次数比文件的更改次数多,那么Access时间就会被频繁更改,更改频率太高了;而Access也算文件的属性,Chang记录的是文件内容属性最新一次被修改时间,那么Access被修改了,文件属性就会变化,Chang就会变化。而文件属性是有大小的,而更改Access时间就需要访问磁盘,那么频繁更改Access时间会频繁访问磁盘,效率降低,占用内存增大。
所以对Access时间进行了优化,Access只有在进行访问文件到达一定次数后才更改Access时间。
那么进行mycode.cpp更改后,Modify和Chang时间都会更改。
那么也可以解释为什么.PHONY修饰下的mycode可以一直执行了,因为它不会对比时间,而是直接执行。
- Makefile的推导规则
结合编译链接的知识点,更改mycode:mycode.cpp的写法:
所以Makefile的推导规则是往下查找,直到找到存在的文件,类似于VS下的调用堆栈,会一直调用直到底层函数,然后一个一个返回。
不过建议直接写成
mycode:mycode.cpp
g++ mycode.cpp -o mycode
补充:make默认从上往下下扫描,第一个命令可以省略名字,make只生成一个可执行程序,默认为第一个。
4. 总结
make是如何工作的,在默认的方式下,也就是我们只输入make命令。那么,
5. 项目清理
- 工程是需要被清理的
- 像clean这种,没有被第一个目标文件直接或间接关联,那么它后面所定义的命令将不会被自动执行,不过,我们可以显示要make执行。即命令——“make clean”,以此来清除所有的目标文件,以便重编译。
- 但是一般我们这种clean的目标文件,我们将它设置为伪目标,用 .PHONY 修饰,伪目标的特性是,总是被执行的。
6. 习题
习题一
1.下列关于makefile描述正确的有?
A.makefile文件保存了编译器和连接器的参数选项
B.主要包含了五个东西:显式规则、隐晦规则、变量定义、文件指示和注释
C.默认的情况下,make命令会在当前目录下按顺序找寻文件名为“GNUmakefile”、“makefile”、“Makefile”的文件, 找到了解释这个文件
D.在Makefile不可以使用include关键字把别的Makefile包含进来
答案:ABC
习题二
2.下列关于make/Makefile描述正确的有?
A.make会生成Makefile中定义的所有目标对象
B.make会自动根据依赖对象检测目标对象是否需要重新生成
C.Makefile中伪对象的功能是目标对象存在则不需要生成
D.Makefile中声明伪对象使用 .PHONY
答案:BD
二.第一个小程序-进度条
在c语言中’\n’的功能相当于’\r’+‘\n’
C语言中的 ‘\n’ 的作用是 回车 + 换行,而不仅仅是换行,这是沿用老式键盘。
1.行缓冲区
我们知道从键盘输入的字符以及向显示器输出的内容,并不会直接读入或输出,而是会先被存放到输入缓冲区与输出缓冲区中,待缓冲区刷新时数据才会才会被读入或输出;而行缓冲是缓冲区类型的一种,在行缓冲下,当 在输入和输出中遇到换行符时,才执行真正的I/O操作;即我们输入的字符会先存放在缓冲区,等按下回车键时才进行真正的I/O操作。
#include<stdio.h>
#include<unistd.h>
int main()
{
printf("you can see me .......");
sleep(5);
return 0;
}
#include<stdio.h>
#include<unistd.h>
int main()
{
printf("you can see me .......");
fflush(stdout);
sleep(5);
return 0;
}
2.倒计时
#include<stdio.h>
#include<unistd.h>
int main()
{
int cnt = 10;
while(cnt)
{
printf("剩余时间:%2d\r",cnt);
cnt--;
fflush(stdout);
sleep(1);
}
return 0;
}
3.进度条
创建的文件:
//Progress.h
#pragma once
#include<stdio.h>
#include<unistd.h>
#include<string.h>
#define NUM 101
#define S_NUM 5
extern void ProccessOn();//函数声明
//Progress.c
#include"Progress.h"
char stype[S_NUM] = {'-','.','#','>','+'};
void ProccessOn()
{
char bar[NUM];
memset(bar,'\0',sizeof(bar));
const char* lable = "|\\-/";
int cnt = 0;//循环101次
while(cnt <= 100)
{
printf("[%-100s][%-3d%%][%c]\r",bar,cnt,lable[cnt%4]);
fflush(stdout);
bar[cnt++] = stype[N];
//sleep(1);
usleep(50000);
}
printf("\n");
}
//main.c
#include"Progress.h"
int main()
{
ProccessOn();
return 0;
}
//Makefile
ProccessOn:main.c Progress.c
gcc -o ProccessOn main.c Progress.c -DN=3//利用命令行传值,选择进度条的形状
.PHONY:clean
clean:
rm -f ProccessOn