makfile编写之万能模板(三)

之前一直没有整理过makefile的使用,现在整理一下便于以后查找。

0、makefile中的常用符号

$@ 表示目标

$^ 表示所有的依赖

$< 表示第一个依赖

$?  --代表当前目标所依赖的文件列表中比当前目标文件还要新的文件。

$*  --不包括后缀名的当前依赖文件的名字

:= 即时赋值

?= 延时赋值

+= 附加

AR   函数库打包程序,可创建静态库.a文档。默认是"ar"。
AS   汇编程序。默认是"as"。
CC   C编译程序。默认是"cc"。
CXX  C++编译程序。默认是"g++"。
CPP  C/C++预处理器。默认是"$(CC) -E"。
FC   Fortran编译器。默认是"f77"。
PC   Pascal语言编译器。默认是"pc"。
YACC Yacc文法分析器。默认是"yacc"。

ARFLAGS     函数库打包程序的命令行参数。默认值是"rv"。
ASFLAGS     汇编程序的命令行参数。
CFLAGS      C编译程序的命令行参数。
CXXFLAGS    C++编译程序的命令行参数。
CPPFLAGS    C/C++预处理器的命令行参数。
FFLAGS      Fortran编译器的命令行参数。
PFLAGS      Pascal编译器的命令行参数。
YFLAGS      Yacc文法分析器的命令行参数。
LDFLAGS     链接器的命令行参数。
LIBS:告诉链接器要链接哪些库文件,如LIBS = -lpthread -liconv

 

1、模板1

ROOTDIR = .
OBJS1 = $(patsubst %.c,%.o,$(wildcard *.c))
OBJS2 = $(patsubst %.cpp,%.o,$(wildcard *.cpp))
DEPE1 = ${OBJS1:%.o=%.d}
DEPE2 = ${OBJS2:%.o=%.d}
CC = arm-linux-gnueabihf-gcc
CXX = arm-linux-gnueabihf-g++
INC = 
LIBS = 
TARGET = test

#INC、LIBS、TARGET根据实际情况填写
#INC为工程头文件目录,如不需额外指定,保持空值即可
#LIBS为库文件目录,如不需额外指定,保持空值即可
#TARGET为生成的目标文件名,必须指定 


$(TARGET) : $(OBJS1) $(OBJS2) 
	$(CXX)  $^ -o $@ $(LIBS) -pthread 

$(OBJS1):%.o:%.c 
	$(CXX) -c $< -o $@ $(INC)
 
$(OBJS2):%.o:%.cpp 
	$(CXX) -c $< -o $@ $(INC)

$(DEPE1):%.d:%.c
	@set -e;rm -f $@;$(CXX) -MM $< $(INC) > $@.$$$$; \
	sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \
	rm -f $@.$$$$

$(DEPE2):%.d:%.cpp
	@set -e;rm -f $@;$(CXX) -MM $< $(INC) > $@.$$$$; \
	sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \
	rm -f $@.$$$$

-include $(DEPE1) $(DEPE2)

.PHONY: clean
clean :
	rm -f $(OBJS1) $(OBJS2)  $(DEPE1) $(DEPE2) $(TARGET)

 

2、模板2

CC=gcc
SRCS=$(wildcard *.c */*.c)
OBJS=$(patsubst %.c, %.o, $(SRCS))
FLAG=-g
NAME=$(wildcard *.c)
TARGET=$(patsubst %.c, %, $(NAME))

$(TARGET):$(OBJS)
	$(CC) -o $@ $^ $(FLAG)
	
%.o:%.c
	$(CC) -o $@ -c $< -g

.PHONY: clean
clean:
	rm -rf $(TARGET) $(OBJS)

解释一下:
  (1)第1行指定编译器为 gcc,可以根据需要修改为 g++ 或者 arm-linux-gcc 等交叉编译工具链。
  (2)第2行是为了获取匹配模式的文件名,*.c 表示当前工程目录的 c 文件,*/*.c 表示所有下一级目录的 .c 文件。
  (3)第3行是将 $(SRCS) 中的 .c 文件替换成 .o 文件。
  (4)第4行用于指定编译选项,根据需要添加,比如 -g、-ml、-Wall、-O2 等等。
  (5)第5、6行,同样使用了 wildcard 和 patsubst 函数来处理,目的是为了指定最终生成的可执行文件名。
  (6)第8行 $(TARGET):$(OBJS) 表示由 .o 文件链接成可执行文件。
  (7)注意第9行前面是一个 <tab> 键,而 $@ 表示目标,也就是 $(TARGET)$^ 表示依赖列表,也就是 $(OBJS) 。
  (8)第11行 %.o:%.c 表示由 .c 文件生成 .o 文件。
  (9)第12行中的 $< 表示搜索到的第一个匹配的文件。
  (10)第14行的 clean 是伪目标(.PHONY),它不需要依赖,执行 make 命令时是不会执行的,当执行 make clean 时才会执行。
  
  另外,Makefile 的赋值除了使用等号(=),还可以使用 := 、+= 、?= 符号。具体含义如下:

  • := 表示简单赋值(注意右边的 $ 值只会向上寻找)
  • += 表示追加赋值
  • ?= 表示判断赋值(判断左边是否已经定义过,如果是则忽略本次赋值)

3、模板3

CC = g++
 
SRCS = $(wildcard *.cpp)
 
OBJS = $(patsubst %cpp, %o, $(SRCS))
 
LDFLAGS :=-lsqlite3 -lTR3A
 
CFLAGS := -Wall -O2 -g -L/home/usr
 
INCLUDE = -I ./com/inc \
        -I ./com/lib/inc
CFLAGS += $(INCLUDE)
 
TARGET = TEST
 
all : $(TARGET)
 
$(TARGET):$(OBJS)
    $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS) 
%.o:%.c
    $(CC) $(CFLAGS) -o $@ -c $< $(LDFLAGS)
 
 
解释(更新中):
    $(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS)   
    (1)-o $@ 与 $^各为一个整体,语句不存在-c,就决定了这行命令是编译,链接过程,
-o后面的$@代表链接后生成的可执行文件的取名,你可以TARGET = main.o,但是main.o代
表的是可执行文件,只是后缀名为.o的可执行文件,而不是编译生成的.o文件,你也可以去掉
-o $@,则链接器自动生成名为a.out的可执行文件。
    $(CC) $(CFLAGS) -o $@ -c $< $(LDFLAGS)  
    (2)-o $@ 与 -c $<各为一个整体,语句中有-c存在,就决定了这行命令只编译不链接,
-o $@代表指定编译生成的.o按照$@取名,而$@本身代表编译后生成的.o,所以可以省略-o $@。
 
疑问:关于-o -c $(CFLAGS)在命令中的位置关系
    如下三种情况等价
    (1)$(CC) $(CFLAGS) -o $@ $^ $(LDFLAGS)
    (2)$(CC) $(CFLAGS) $^ -o $@ $(LDFLAGS)
    (3)$(CC) $^ -o $@ $(LDFLAGS) $(CFLAGS) 
 
 
    $(CC) $(CFLAGS) -o $@ -c $< $(LDFLAGS) 等价于 $(CC) $(CFLAGS) -o $@ -c $<
 
 
 

4、生成可执行文件的makefile

######################################
#
######################################
#source file
#源文件,自动找所有.c和.cpp文件,并将目标定义为同名.o文件
SOURCE  := $(wildcard *.c) $(wildcard *.cpp)
OBJS    := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
  
#target you can change test to what you want
#目标文件名,输入任意你想要的执行文件名
TARGET  := test
  
#compile and lib parameter
#编译参数
CC      := gcc
LIBS    :=
LDFLAGS :=
DEFINES :=
INCLUDE := -I.
CFLAGS  := -g -Wall -O3 $(DEFINES) $(INCLUDE)
CXXFLAGS:= $(CFLAGS) -DHAVE_CONFIG_H
  
  
#i think you should do anything here
#下面的基本上不需要做任何改动了
.PHONY : everything objs clean veryclean rebuild
  
everything : $(TARGET)
  
all : $(TARGET)
  
objs : $(OBJS)
  
rebuild: veryclean everything
                
clean :
    rm -fr *.so
    rm -fr *.o
    
veryclean : clean
    rm -fr $(TARGET)
  
$(TARGET) : $(OBJS)
    $(CC) $(CXXFLAGS) -o $@ $(OBJS) $(LDFLAGS) $(LIBS)

 

5、生成静态链接库的makefile

######################################
#
#
######################################
  
#target you can change test to what you want
#共享库文件名,lib*.a
TARGET  := libtest.a
  
#compile and lib parameter
#编译参数
CC      := gcc
AR      = ar
RANLIB  = ranlib
LIBS    :=
LDFLAGS :=
DEFINES :=
INCLUDE := -I.
CFLAGS  := -g -Wall -O3 $(DEFINES) $(INCLUDE)
CXXFLAGS:= $(CFLAGS) -DHAVE_CONFIG_H
  
#i think you should do anything here
#下面的基本上不需要做任何改动了
  
#source file
#源文件,自动找所有.c和.cpp文件,并将目标定义为同名.o文件
SOURCE  := $(wildcard *.c) $(wildcard *.cpp)
OBJS    := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
  
.PHONY : everything objs clean veryclean rebuild
  
everything : $(TARGET)
  
all : $(TARGET)
  
objs : $(OBJS)
  
rebuild: veryclean everything
                
clean :
    rm -fr *.o
    
veryclean : clean
    rm -fr $(TARGET)
  
$(TARGET) : $(OBJS)
    $(AR) cru $(TARGET) $(OBJS)
    $(RANLIB) $(TARGET)

 

6、生成动态链接库的makefile

######################################
#
#
######################################
  
#target you can change test to what you want
#共享库文件名,lib*.so
TARGET  := libtest.so
  
#compile and lib parameter
#编译参数
CC      := gcc
LIBS    :=
LDFLAGS :=
DEFINES :=
INCLUDE := -I.
CFLAGS  := -g -Wall -O3 $(DEFINES) $(INCLUDE)
CXXFLAGS:= $(CFLAGS) -DHAVE_CONFIG_H
SHARE   := -fPIC -shared -o
  
#i think you should do anything here
#下面的基本上不需要做任何改动了
  
#source file
#源文件,自动找所有.c和.cpp文件,并将目标定义为同名.o文件
SOURCE  := $(wildcard *.c) $(wildcard *.cpp)
OBJS    := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
  
.PHONY : everything objs clean veryclean rebuild
  
everything : $(TARGET)
  
all : $(TARGET)
  
objs : $(OBJS)
  
rebuild: veryclean everything
                
clean :
    rm -fr *.o
    
veryclean : clean
    rm -fr $(TARGET)
  
$(TARGET) : $(OBJS)
    $(CC) $(CXXFLAGS) $(SHARE) $@ $(OBJS) $(LDFLAGS) $(LIBS)

 

7、makefile中添加安装与卸载

install:
	mkdir -p $(OUTPUT)/include/algorithm
	cp $(TGT_LIB_H) $(OUTPUT)/include/algorithm/
	cp $(TGT_LIB_A) $(OUTPUT)/lib/
	cp $(TGT_LIB_SO) $(OUTPUT)/lib/
	
uninstall:
	rm -rf $(OUTPUT)/include/algorithm
	rm -f $(OUTPUT)/lib/$(TGT_LIB_A)
	rm -f $(OUTPUT)/lib/$(TGT_LIB_SO)

8、makefile万能模板

######################################
# Generic makefile
######################################
# 用户设定
# 如果需要,调整下面的东西。 EXECUTABLE 是目标的可执行文件名, LIBS
# 是一个需要连接的程序包列表(例如 alleg, stdcx, iostr 等等)。当然你
# 可以在 make 的命令行覆盖它们,你愿意就没问题。
EXECUTABLE :=../../obj/xxxxx.so
LIBS=../lib/centos-5.4-64/libx1.a ../lib/libx2.a ../lib/libx3.a ../lib/libx4.a ../lib/libx5.a ../lib/libx6.a  ../lib/libx7.a ../lib/libiconv.a  ../lib/libjasper.a
# 现在来改变任何你想改动的隐含规则中的变量,例如
#CFLAGS := -g -Wall -O -I /usr/lib/jvm/jdk1.8.0_05/include/linux/ -I /usr/lib/jvm/jdk1.8.0_05/include/
CFLAGS := -O2  -I /usr/lib/jdk1.7.0_79/include/linux/ -I /usr/lib/jdk1.7.0_79/include/ -fPIC -m64 
CXXFLAGS := $(CFLAGS)
# 下面先检查你的 djgpp 命令目录下有没有 rm 命令,如果没有,我们使用
# del 命令来代替,但有可能给我们 \'File not found\' 这个错误信息,这没
# 什么大碍。如果你不是用 DOS ,把它设定成一个删文件而不废话的命令。
# (其实这一步在 UNIX 类的系统上是多余的,只是方便 DOS 用户。 UNIX
# 用户可以删除这5行命令。)
#ifneq ($(wildcard $(DJDIR)/bin/rm.exe),)
#RM-F := rm -f
#else
#RM-F := del
#endif
# 从这里开始,你应该不需要改动任何东西。
SOURCE := $(wildcard *.c) $(wildcard *.cpp)
OBJS := $(patsubst %.c,%.o,$(patsubst %.cpp,%.o,$(SOURCE)))
DEPS := $(patsubst %.o,%.d,$(OBJS))
MISSING_DEPS := $(filter-out $(wildcard $(DEPS)),$(DEPS))
MISSING_DEPS_SOURCES := $(wildcard $(patsubst %.d,%.c,$(MISSING_DEPS)) \
$(patsubst %.d,%.cpp,$(MISSING_DEPS)))
CPPFLAGS +=-MD
.PHONY : everything deps objs clean veryclean rebuild dllLib
everything : $(EXECUTABLE)
deps : $(DEPS)
objs : $(OBJS)
clean :
	@$(RM) *.o
	@$(RM) *.d
veryclean: clean
	@$(RM) $(EXECUTABLE)
	cd ../libjasper/libjasper && $(MAKE) -f libjasperMakefile64 veryclean
	cd ../libiconv-1.14/libiconv && $(MAKE) -f libiconvMakefile64 veryclean
	cd ../x2 && $(MAKE) -f x2Makefile64 veryclean
	cd ../x3 && $(MAKE) -f x3Makefile64 veryclean
	cd ../x4 && $(MAKE) -f x4 veryclean
	cd ../x1 && $(MAKE) -f x1MakefileASmart64 veryclean
rebuild: veryclean everything
ifneq ($(MISSING_DEPS),)
$(MISSING_DEPS) :
	@$(RM) $(patsubst %.d,%.o,$@)
endif
-include $(DEPS)
$(EXECUTABLE) : $(OBJS) $(LIBS)
	g++  -shared  -o  $(EXECUTABLE) $(OBJS) $(LIBS) -l pthread
$(LIBS):dllLib
	cd ../libjasper/libjasper && $(MAKE) -f libjasperMakefile64
	cd ../libiconv-1.14/libiconv && $(MAKE) -f libiconvMakefile64
	cd ../x2 && $(MAKE) -f x2Makefile64
	cd ../x3 && $(MAKE) -f x3Makefile64
	cd ../x4 && $(MAKE) -f x4ImageMakefile64
	cd ../x1 && $(MAKE) -f x1MakefileASmart64
dllLib:;

 

参考:Makefile常用万能模板(包括静态链接库、动态链接库、可执行文件)

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

落花逐流水

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值