Linux MakeFile (编写、函数以及指定位置存放)

来源:互联网 发布:mac地址 编辑:程序博客网 时间:2024/06/03 23:38

1、MakeFile 编写

正如前面所说的,如果一个工程有3个头文件,和8个C文件,我们为了完成前面所述的那三个规则,我们的Makefile应该是下面的这个样子的。

edit : main.o kbd.o command.o display.o \insert.o search.o files.o utils.occ -o edit main.o kbd.o command.o display.o \insert.o search.o files.o utils.omain.o : main.c defs.hcc -c main.ckbd.o : kbd.c defs.h command.hcc -c kbd.ccommand.o : command.c defs.h command.hcc -c command.cdisplay.o : display.c defs.h buffer.hcc -c display.cinsert.o : insert.c defs.h buffer.hcc -c insert.csearch.o : search.c defs.h buffer.hcc -c search.cfiles.o : files.c defs.h buffer.h command.hcc -c files.cutils.o : utils.c defs.hcc -c utils.cclean :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.oedit : $(objects)cc -o edit $(objects)main.o : main.c defs.hcc -c main.ckbd.o : kbd.c defs.h command.hcc -c kbd.ccommand.o : command.c defs.h command.hcc -c command.cdisplay.o : display.c defs.h buffer.hcc -c display.cinsert.o : insert.c defs.h buffer.hcc -c insert.csearch.o : search.c defs.h buffer.hcc -c search.cfiles.o : files.c defs.h buffer.h command.hcc -c files.cutils.o : utils.c defs.hcc -c utils.cclean :rm edit $(objects)

  • 第二次优化

试想如果我们的“%.o”有几百个,此时上述方法肯定很繁琐,下述方法是常用的。

CFLAGS=  LIBS = CC=gccproject=outputNameobjects=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:commtestCFLAGS=-fPIC -g -WallARIA_INCLUDE=-I/usr/local/Aria/includeARIA_LINK=-L/usr/local/Aria/lib -lAria -lpthread -ldl -lrt%: %.cpp    $(CXX) $(CFLAGS) $(ARIA_INCLUDE) $< -o $@ $(ARIA_LINK).PHONY:cleanclean:    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=gccOBJDIR=./dirproject=$(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:cleanclean:  rm *~ *.o -f $(project) 

倘若这个目录不存在呢?这个时候应该先创建这个目录,再编译



1 0
原创粉丝点击