• 【Linux】Linux下的自动化构建工具——make/makefile



    需要云服务器等云产品来学习Linux的同学可以移步/-->腾讯云<--/-->阿里云<--/-->华为云<--/官网,轻量型云服务器低至112元/年,新用户首次下单享超低折扣。 


    目录

    一、make和makefile的区别

    二、makefile的原理

    1、make的原理详解

    2、临时文件的清理

    3、文件的三个时间

    三、makefile的推导规则

    四、实现一个进度条

    1、缓冲区问题

    2、实现倒计时程序

    3、进度条的实现


    一、make和makefile的区别

    make是一个命令,makefile是一个文件。

    二、makefile的原理

    1、makefile的原理详解

    每次生成可执行文件时需要输入一长串的gcc········指令是不是感觉很麻烦?makefile存在的意义是为了项目的构建。需要表明依赖关系和依赖方法。

    2、临时文件的清理

    1. mycode:mycode.c
    2. gcc mycode.c -o mycode -g
    3. .PHONY:clean
    4. clean:
    5. rm -f mycode

    -g代表可被调试。 

    1. mycode:mycode.c
    2. gcc -o $@ $^
    3. .PHONY:clean
    4. clean:
    5. rm -f mycode

    $@代表生成的目标文件mycode,$^代表依赖文件mycode.c 。

    .PHONY表示被该关键字修饰的对象是一个伪目标。(该伪目标总是可被执行)

    我们在编写完makefile后,make一下,就能生成对应的可执行程序。如果依赖的文件列表没有发生修改,gcc通过判断mycode.c的Modify time早于mycode的Modify time,说明依赖文件列表没有修改,那么再次make将会失败;反之,重新编译将会成功。

    不过多次make clean却不会失败,因为clean被.PHONY修饰,变成了一个伪目标,所以clean总是可被执行。

    .c位于不同目录写法,生成.o文件:

    1. obj=mycode.o add.o sub.o
    2. mycode:$(obj)
    3. gcc -o $@ $^
    4. %.o:%.c
    5. gcc -c $<
    6. %.o:./test_lib/%.c
    7. gcc -c $<
    8. .PHONY:clean
    9. clean:
    10. rm -f *.o mycode删掉所有.o和mycode

    %是通配符,%.o表示将当前目录下的.o目标文件全部展开。

    $<表示将%.c展开,一个一个进行编译,上方代码没有指明-o,将会形成同名.o。

    3、文件的三个时间

    1. [root@VM-4-11-centos ~]# stat 106/test.c
    2. File: ‘106/test.c’
    3. Size: 0 Blocks: 0 IO Block: 4096 regular empty file
    4. Device: fd01h/64769d Inode: 658416 Links: 1
    5. Access: (0644/-rw-r--r--) Uid: ( 0/ root) Gid: ( 0/ root)
    6. Access: 2022-09-17 21:13:21.098656019 +0800//最近访问
    7. Modify: 2022-09-17 21:13:21.098656019 +0800//最近修改
    8. Change: 2022-09-17 21:13:21.098656019 +0800//最近改动
    9. Birth: -

    1、读取文件时,Access time被改变,而Modify、Change time不会被改变(为防止文件被频繁访问修改Access time,优化为多次访问才修改一次该时间)

    2、修改文件时,三个时间都会被改变

    3、修改属性时,Change time被改变,而Access、Modify time不会被改变

    三、makefile的推导规则

    makefile会自己逐步寻找依赖文件列表,如果没找到对于依赖文件,将会继续向下寻找该依赖文件的依赖方法。

    四、实现一个进度条

    1、缓冲区问题

    1. #include
    2. #include
    3. int main()
    4. {
    5. printf("hello world\n");//如果这里没有\n,程序将会在3秒后执行打印
    6. //fflush(stdout);//fflush会立即刷新缓冲区
    7. sleep(3);
    8. return 0;
    9. }

    因为\n才会刷新缓冲区(\r等不会),这个程序printf执行完之后,将会休眠3秒,如果没有\n或者fflush手动刷新缓冲区,hello world将会存放至缓冲区,程序结束后才执行打印。

    2、实现倒计时程序

    1. #include
    2. #include
    3. int main()
    4. {
    5. int cnt=10;
    6. while(cnt)
    7. {
    8. printf("%2d\r",cnt);//2d控制刷新两位,\r表示回车
    9. fflush(stdout);//手动刷新缓冲区
    10. sleep(1);
    11. --cnt;
    12. }
    13. return 0;
    14. }

    能够实现从10到计时的程序。

    3、进度条的实现

    1. //.h文件
    2. #pragma once
    3. #include
    4. #include
    5. #include
    6. #define NUM 101
    7. #define STYLE '*'
    8. extern void processOn();
    9. //.c文件
    10. #include "process.h"
    11. void processOn()
    12. {
    13. int cnt=0;
    14. char bar[NUM];
    15. memset(bar,'\0',NUM);
    16. char type[4]={'|','/','-','\\'};
    17. while(cnt<=100)
    18. {
    19. //-100表示占据屏幕100个空间并左对齐,%%转义为%
    20. printf("[%-100s][%d%%][%c]\r",bar,cnt,type[cnt%4]);
    21. bar[cnt++]=STYLE;
    22. fflush(stdout);
    23. usleep(50000);
    24. }
    25. printf("\n");
    26. }

    makefile中不用写头文件,编译器自己会去找。

    <>查找方式:直接去库目录下查找,如果找不到,报编译错误;

    ""查找方式:先去代码所在的路径下查找,如果找不到,再去库目录下查找,如果找不到,报编译错误。

  • 相关阅读:
    《Kubernetes部署篇:Ubuntu20.04基于containerd二进制部署K8S 1.25.14集群(多主多从)》
    在 Wed 中应用 MyBatis(同时使用MVC架构模式,以及ThreadLocal 事务控制)
    【学习笔记】字符串
    JAVAEE—HTTP
    day4 力扣 11.22 素数个数统计
    你不知道的CSS 伪元素
    Spring Cloud(五):Spring Cloud Alibaba Nacos 1.4.X 注册中心AP & CP架构Raft源码分析
    sleuth链路追踪
    【漏洞复现】fastjson_1.2.24_unserializer_rce
    OpenGL编程(一):三角形大作
  • 原文地址:https://blog.csdn.net/gfdxx/article/details/127564396