1、MakeFile 编写
正如前面所说的,如果一个工程有3个头文件,和8个C文件,我们为了完成前面所述的那三个规则,我们的Makefile应该是下面的这个样子的。
edit : main.o kbd.o command.o display.o \
insert.o search.o files.o utils.o
cc -o edit main.o kbd.o command.o display.o \
insert.o search.o files.o utils.o
main.o : main.c defs.h
cc -c main.c
kbd.o : kbd.c defs.h command.h
cc -c kbd.c
command.o : command.c defs.h command.h
cc -c command.c
display.o : display.c defs.h buffer.h
cc -c display.c
insert.o : insert.c defs.h buffer.h
cc -c insert.c
search.o : search.c defs.h buffer.h
cc -c search.c
files.o : files.c defs.h buffer.h command.h
cc -c files.c
utils.o : utils.c defs.h
cc -c utils.c
clean :
rm edit main.o kbd.o command.o display.o \
insert.o search.o files.o utils.o
反斜杠(\)是换行符的意思。这样比较便于Makefile的易读。我们可以把这个内容保存在文件为“Makefile”或“makefile”的文件中,然后在该目录下直接输入命令“make”就可
以生成执行文件edit。如果要删除执行文件和所有的中间目标文件,那么,只要简单地执行一下“make clean”就可以了。
- 第一次优化
本段 Makefile 用到了三处啊 main.o ..,故用一个变量去代替
objects = main.o kbd.o command.o display.o \
insert.o search.o files.o utils.o
edit : $(objects)
cc -o edit $(objects)
main.o : main.c defs.h
cc -c main.c
kbd.o : kbd.c defs.h command.h
cc -c kbd.c
command.o : command.c defs.h command.h
cc -c command.c
display.o : display.c defs.h buffer.h
cc -c display.c
insert.o : insert.c defs.h buffer.h
cc -c insert.c
search.o : search.c defs.h buffer.h
cc -c search.c
files.o : files.c defs.h buffer.h command.h
cc -c files.c
utils.o : utils.c defs.h
cc -c utils.c
clean :
rm edit $(objects)
- 第二次优化
试想如果我们的“%.o”有几百个,此时上述方法肯定很繁琐,下述方法是常用的。
CFLAGS=
LIBS =
CC=gcc
project=outputName
objects=main.o kbd.o command.o display.o \
insert.o search.o files.o utils.o
$(project):$(objects)
$(CC) $< -o $@
$(objects):%.o:%.c
$(CC) $(CFLAGS) -c $< $(LIBS)
clean :
rm $(objects) $(project)
这个是一个简洁版的:
all:commtest
CFLAGS=-fPIC -g -Wall
ARIA_INCLUDE=-I/usr/local/Aria/include
ARIA_LINK=-L/usr/local/Aria/lib -lAria -lpthread -ldl -lrt
%: %.cpp
$(CXX) $(CFLAGS) $(ARIA_INCLUDE) $< -o $@ $(ARIA_LINK)
.PHONY:clean
clean:
rm -f commtest *.o
2、MakeFile 函数
在Makefile规则中,通配符会被自动展开。但在变量的定义和函数引用时,通配符将失效。这种情况下如果需要通配符有效,就需要使用函数“wildcard”,它的用法是:$(wildcard PATTERN...) 。在Makefile中,它被展开为已经存在的、使用空格分开的、匹配此模式的所有文件列表。如果不存在任何符合此模式的文件,函数会忽略模式字符并返回空。需要注意的是:这种情况下规则中通配符的展开和上一小节匹配通配符的区别。
一般我们可以使用“$(wildcard *.c)”来获取工作目录下的所有的.c文件列表。复杂一些用法;可以使用“$(patsubst %.c,%.o,$(wildcard *.c))”,首先使用“wildcard”函数获取工作目录下的.c文件列表;之后将列表中所有文件名的后缀.c替换为.o。这样我们就可以得到在当前目录可生成的.o文件列表。因此在一个目录下可以使用如下内容的Makefile来将工作目录下的所有的.c文件进行编译并最后连接成为一个可执行文件:
CFLAGS=
LIBS =
C_SRC = $(wildcard *.c)
C_OBJ = $(patsubst %c, %o, $(C_SRC))
CPP_SRC = $(wildcard *.cpp)
CPP_OBJ = $(patsubst %cpp, %o, $(CPP_SRC))
.PHONY:all clean
all:$(CPP_OBJ) $(C_OBJ)
%.o:%.c
gcc $(CFLAGS) -o $@ $< $(LIBS)
%.o:%.c
g++ $(CFLAGS) -o $@ $< $(LIBS)
clean:
rm *~ *.o -f
下面我们要介绍的函数主要是处理文件名的。每个函数的参数字符串都会被当做一个或是一系列的文件名来对待。
$(dir ) 名称:取目录函数——dir。功能:从文件名序列中取出目录部分。目录部分是指最后一个反斜杠(“/”)之前的部分。如果没有反斜杠,那么返回“./”。返回:返回文件名序列的目录部分。示例: $(dir src/foo.c hacks)返回值是“src/ ./”。
$(notdir ) 名称:取文件函数——notdir。功能:从文件名序列中取出非目录部分。非目录部分是指最后一个反斜杠(“/”)之后的部分。返回:返回文件名序列的非目录部分。示例: $(notdir src/foo.c hacks)返回值是“foo.c hacks”。
$(suffix ) 名称:取后缀函数——suffix。功能:从文件名序列中取出各个文件名的后缀。返回:返回文件名序列的后缀序列,如果文件没有后缀,则返回空字串。示例:$(suffix src/foo.c src-1.0/bar.c hacks)返回值是“.c .c”。
$(basename )名称:取前缀函数——basename。功能:从文件名序列中取出各个文件名的前缀部分。返回:返回文件名序列的前缀序列,如果文件没有前缀,则返回空字串。示例:$(basename src/foo.c src-1.0/bar.c hacks)返回值是“src/foo src-1.0/bar hacks”。
$(addsuffix , ) 名称:加后缀函数——addsuffix。功能:把后缀加到中的每个单词后面。返回:返回加过后缀的文件名序列。示例:$(addsuffix .c,foo bar)返回值是“foo.c bar.c”。
$(addprefix , ) 名称:加前缀函数——addprefix。功能:把前缀加到中的每个单词后面。返回:返回加过前缀的文件名序列。示例:$(addprefix src/,foo bar)返回值是“src/foo src/bar”。
$(join , )名称:连接函数——join。功能:把中的单词对应地加到的单词后面。如果的单词个数要比的多,那么,中的多出来的单词将保持原样。如果的单词个数要比多,那么,多出来的单词将被复制到中。返回:返回连接过后的字符串。示例:$(join aaa bbb , 111 222 333)返回值是“aaa111 bbb222 333”。
3、指定输出文件存放位置
最常用的方法就是,需要先创建一个目录去存储
CFLAGS=
LIBS =
CC=gcc
OBJDIR=./dir
project=$(OBJDIR)/outputName
C_SRC = $(wildcard *.c)
C_OBJ = $(patsubst %c, %o, $(C_SRC))
CPP_SRC = $(wildcard *.cpp)
CPP_OBJ = $(patsubst %cpp, %o, $(CPP_SRC))
$(project):$(CPP_OBJ) $(C_OBJ)
$(CC) $(CFLAGS) -o $@ $< $(LIBS)
%.o:%.c
$(CC) $(CFLAGS) -c $< $(LIBS)
%.o:%.cpp
$(CC)++ $(CFLAGS) -o $< $(LIBS)
.PHONY:clean
clean:
rm *~ *.o -f $(project)
倘若这个目录不存在呢?这个时候应该先创建这个目录,再编译