江山市建设局网站网络营销环境的分析主要是
makefile简易教程

一、学习目标
达到多文件快速编译的需求,相关符号的意思,以及其它注意事项。
二、快速入门
2.1 基本概念
Makefile 是一个在Unix和Linux操作系统上使用的构建工具,用于自动化编译和构建源代码。
 
2.2 用处
通过Makefile,你可以定义一系列规则来指示如何构建源代码,以及定义每个规则所需要的依赖关系。Makefile也可以在构建过程中执行自定义命令,例如清理旧的构建输出、运行测试等。
2.3 基本语法
目标文件:依赖文件指令
target: dependenciescommand
 
其中,target 表示构建目标(例如可执行文件、库文件、对象文件等),dependencies 表示构建目标所依赖的文件(可以是其他目标或源代码文件),command 表示构建目标的命令。注意,command 前面必须有一个tab(制表符)。
例如:
test:test.cgcc -o test test.c
 
2.4 变量
Makefile还支持变量,用于在不同规则之间共享常量或者路径等信息。变量的定义格式如下:
VARIABLE_NAME = variable_value
 
示例:
CC = gcc
CFLAGS = -Wall -Werrorhello: hello.c$(CC) $(CFLAGS) -o hello hello.c
 
在这个示例中,CC 变量定义了编译器的名称,CFLAGS 变量定义了编译器的参数。在构建hello目标时,Makefile使用这些变量来执行编译命令。
2.5 构建多个目标

当有多个.c文件需要进行编译时,可以这样构建makefile
all: hello worldhello: hello.c$(CC) $(CFLAGS) -o hello hello.cworld: world.c$(CC) $(CFLAGS) -o world world.c
 
在这个示例中,all 是一个伪目标,它依赖于hello和world。当我们运行make all时,Make会自动构建hello和world目标。
注意,all是一个伪目标,它并不会生成任何文件。在Makefile中,有一些特殊的目标(例如all、clean等)通常被定义为伪目标。
2.6 清理目标
Makefile还可以定义清理目标,用于删除构建过程中生成的临时文件。例如:
clean:rm -f hello world *.o
 
在这个示例中,clean 是一个伪目标,它的命令会删除当前目录下的hello、world和所有.o文件。
2.7 .PHONY

在Makefile中,.PHONY是一个特殊的目标,用于声明一些伪目标(phony target),也就是那些不对应任何实际的文件或目录,只是用来执行一些特定的操作的目标。
声明伪目标的方式很简单,只需要在Makefile中添加以下语句:
.PHONY: target1 target2 ...
 
其中,target1、target2等为伪目标的名称,可以声明多个伪目标,它们之间用空格分隔。
使用.PHONY声明伪目标的好处在于,它可以避免出现一些潜在的问题,例如:
- 当伪目标与实际文件或目录同名时,Makefile可能会将其误认为是一个文件或目录,从而导致错误的行为。使用
.PHONY可以显式地声明这些目标是伪目标,避免这种情况发生。 - 如果在命令行中执行
make target时,target已经存在,并且其时间戳比依赖文件的时间戳更晚,Makefile默认情况下不会重新构建target。但是如果target是一个伪目标,那么即使时间戳较晚,Makefile也会执行伪目标下的命令,因为伪目标并不对应任何实际的文件或目录。 
例如,我们可以在Makefile中声明一个名为clean的伪目标,用于清理项目中生成的临时文件和目录,如下所示:
.PHONY: cleanclean:rm -rf build/* bin/* 
在命令行中执行make clean时,Makefile会执行rm -rf build/* bin/*命令,清理build目录和bin目录中的所有文件。
需要注意的是,.PHONY目标本身并不会执行任何操作,它只是用来声明伪目标。因此,在Makefile中,.PHONY目标通常放在文件的最前面或最后面,以方便查看和维护。
2.8 $符号
在Makefile中,@和@和@和^等符号是自动化变量,它们代表了不同的含义:
- $@ 表示当前规则中的目标文件,即被生成的文件名。
 - $^ 表示当前规则中所有的依赖文件,以空格分隔。
 
例如:
makefileCopy code
app: main.o utils.ogcc $^ -o $@main.o: main.cgcc -c $< -o $@utils.o: utils.cgcc -c $< -o $@
 
在上面的示例中,我们定义了一个目标文件app,它依赖于两个对象文件main.o和utils.o。在生成目标文件时,我们使用将所有依赖文件都链接在一起,使用^将所有依赖文件都链接在一起,使用将所有依赖文件都链接在一起,使用@作为生成的目标文件名。同时,我们还定义了两个规则分别用于生成main.o和utils.o文件,在这里我们使用$<代表当前规则中的第一个依赖文件,即源代码文件名。
2.9 makefile与其它语言冲突
例如我在使用C语言连接MySQL数据库时,使用到的编译选项中含有$()的字样,然而makefile中,这类字样指代某个变量,所以产生冲突,导致无法make成功。
可以将编译选项改为$$(),双美元符号可以提示makefile需要显式调用的。
上述例子如下:
.PHONY:all
all:test_cgi mysql_conntest_cgi:test_cgi.ccg++ -o $@ $^mysql_conn:mysql_conn.ccg++ -o $@ $^ -std=c++11 $$(mysql_config --cflags --libs).PHONY:clean
clean:rm -f test_cgi mysql_conn
 

