文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Makefile及其工作原理是什么

2023-06-28 15:52

关注

这篇文章给大家介绍Makefile及其工作原理是什么,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。

当你需要在一些源文件改变后运行或更新一个任务时,通常会用到 make 工具。make 工具需要读取一个 Makefile(或 makefile)文件,在该文件中定义了一系列需要执行的任务。你可以使用 make 来将源代码编译为可执行程序。大部分开源项目会使用 make 来实现最终的二进制文件的编译,然后使用 make install 命令来执行安装。

依然从打印 “Hello World” 开始。首先创建一个名字为 myproject 的目录,目录下新建 Makefile 文件,文件内容为:

  1. say_hello:

  2. echo"Hello World"

在 myproject 目录下执行 make,会有如下输出:

  1. $ make

  2. echo"Hello World"

  3. HelloWorld

在上面的例子中,“say_hello” 类似于其他编程语言中的函数名。这被称之为目标target。在该目标之后的是预置条件或依赖。为了简单起见,我们在这个示例中没有定义预置条件。echo ‘Hello World' 命令被称为步骤recipe。这些步骤基于预置条件来实现目标。目标、预置条件和步骤共同构成一个规则。

总结一下,一个典型的规则的语法为:

  1. 目标:预置条件

  2. 步骤

作为示例,目标可以是一个基于预置条件(源代码)的二进制文件。另一方面,预置条件也可以是依赖其他预置条件的目标。

  1. final_target: sub_target final_target.c

  2. Recipe_to_create_final_target

  3. sub_target: sub_target.c

  4. Recipe_to_create_sub_target

目标并不要求是一个文件,也可以只是步骤的名字,就如我们的例子中一样。我们称之为“伪目标”。

再回到上面的示例中,当 make 被执行时,整条指令 echo "Hello World" 都被显示出来,之后才是真正的执行结果。如果不希望指令本身被打印处理,需要在 echo 前添加 @。 say_hello:    @echo "Hello World"

重新运行 make,将会只有如下输出:

  1. $ make

  2. HelloWorld

接下来在 Makefile 中添加如下伪目标:generate 和 clean:

  1. say_hello:

  2. @echo"Hello World"

  3. generate:

  4. @echo"Creating empty text files..."

  5. touchfile-{1..10}.txt

  6. clean:

  7. @echo"Cleaning up..."

  8. rm*.txt

随后当我们运行 make 时,只有 say_hello 这个目标被执行。这是因为Makefile 中的第一个目标为默认目标。通常情况下会调用默认目标,这就是你在大多数项目中看到 all 作为第一个目标而出现。all 负责来调用它他的目标。我们可以通过 .DEFAULT_GOAL 这个特殊的伪目标来覆盖掉默认的行为。

在 Makefile 文件开头增加 .DEFAULT_GOAL:

  1. .DEFAULT_GOAL := generate

make 会将 generate 作为默认目标:

  1. $ make

  2. Creatingempty text files...

  3. touchfile-{1..10}.txt

顾名思义,.DEFAULT_GOAL 伪目标仅能定义一个目标。这就是为什么很多 Makefile 会包括 all 这个目标,这样可以调用多个目标。

下面删除掉 .DEFAULT_GOAL,增加 all 目标:

  1. all: say_hello generate

  2. say_hello:

  3. @echo"Hello World"

  4. generate:

  5. @echo"Creating empty text files..."

  6. touchfile-{1..10}.txt

  7. clean:

  8. @echo"Cleaning up..."

  9. rm*.txt

运行之前,我们再增加一些特殊的伪目标。.PHONY 用来定义这些不是文件的目标。make 会默认调用这些伪目标下的步骤,而不去检查文件名是否存在或最后修改日期。完整的 Makefile 如下:

  1. .PHONY: all say_hello generate clean

  2. all: say_hello generate

  3. say_hello:

  4. @echo"Hello World"

  5. generate:

  6. @echo"Creating empty text files..."

  7. touchfile-{1..10}.txt

  8. clean:

  9. @echo"Cleaning up..."

  10. rm*.txt

make 命令会调用 say_hello 和 generate:

  1. $ make

  2. HelloWorld

  3. Creatingempty text files...

  4. touchfile-{1..10}.txt

clean 不应该被放入 all 中,或者被放入第一个目标中。clean 应当在需要清理时手动调用,调用方法为 make clean。

  1. $ make clean

  2. Cleaning up...

  3. rm*.txt

现在你应该已经对 Makefile 有了基础的了解,接下来我们看一些进阶的示例。

进阶示例
变量

在之前的实例中,大部分目标和预置条件是已经固定了的,但在实际项目中,它们通常用变量和模式来代替。

定义变量最简单的方式是使用 = 操作符。例如,将命令 gcc 赋值给变量 CC:

  1. CC =gcc

这被称为递归扩展变量,用于如下所示的规则中:

  1. hello: hello.c

  2. ${CC} hello.c -o hello

你可能已经想到了,这些步骤将会在传递给终端时展开为:

  1. gcc hello.c -o hello

${CC} 和 $(CC) 都能对 gcc 进行引用。但如果一个变量尝试将它本身赋值给自己,将会造成死循环。让我们验证一下:

  1. CC =gcc

  2. CC = ${CC}

  3. all:

  4. @echo ${CC}

此时运行 make 会导致:

  1. $ make

  2. Makefile:8:***Recursive variable 'CC' references itself (eventually). Stop.

为了避免这种情况发生,可以使用 := 操作符(这被称为简单扩展变量)。以下代码不会造成上述问题:

  1. CC :=gcc

  2. CC := ${CC}

  3. all:

  4. @echo ${CC}

模式和函数

下面的 Makefile 使用了变量、模式和函数来实现所有 C 代码的编译。我们来逐行分析下:

  1. #Usage:

  2. #make     # compile all binary

  3. #make clean  # remove ALL binaries and objects

  4. .PHONY = all clean

  5. CC =gcc            # compiler to use

  6. LINKERFLAG =-lm

  7. SRCS := $(wildcard *.c)

  8. BINS := $(SRCS:%.c=%)

  9. all: ${BINS}

  10. %:%.o

  11. @echo"Checking.."

  12. ${CC} ${LINKERFLAG} $

  13. %.o:%.c

  14. @echo"Creating object.."

  15. ${CC}-c $

  16. clean:

  17. @echo"Cleaning up..."

  18. rm-rvf *.o ${BINS}

下面是重写后的 Makefile,该文件应该被放置在一个有 foo.c 文件的目录下:

  1. #Usage:

  2. #make     # compile all binary

  3. #make clean  # remove ALL binaries and objects

  4. .PHONY = all clean

  5. CC =gcc            # compiler to use

  6. LINKERFLAG =-lm

  7. SRCS := foo.c

  8. BINS := foo

  9. all: foo

  10. foo: foo.o

  11. @echo"Checking.."

  12. gcc-lm foo.o -o foo

  13. foo.o: foo.c

  14. @echo"Creating object.."

  15. gcc-c foo.c

  16. clean:

  17. @echo"Cleaning up..."

  18. rm-rvf foo.o foo

关于Makefile及其工作原理是什么就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯