如何写一个makefile

最近因为编译需求,需要更改一些编译条件,顺带看了一些Makefile相关的知识,介绍的很详细,但是例子很少,拆分的比较零碎。初学的话,确实有点压力,我还是喜欢直接在原有的基础上改一些东西,然后遇到问题再进行查找。

GCC := gcc
GXX := g++ SRC_DIR := source
OUT_DIR := build ##依赖的头文件目录
INC_DIR := /usr/include
INC_DIR += ./include ##依赖的库文件目录
LIB_DIR := /usr/lib/debug/linux_x32/lib
LIB_DIR += /usr/lib/debug/linux_x64/lib ##遍历头文件目录中的每一个文件
CFLAGS := $(foreach a, $(INC_DIR), -I$(a)) ##遍历库文件目录中的每一个库
LDFLAGS := $(foreach a, $(LIB_DIR), -L$(a)) ##OBJ(.o)文件名称
OBJS := mcmerge.o ##遍历编译出来的OBJ(.o)文件
FINAL_OBJS := $(foreach obj, $(OBJS), $(OUT_DIR)/$(obj)) ##列举所需要的库文件
LIBS := yuv \
imageunit_linux \
imf \
intlc \
videomanage_linux \
svml ##组装成所需要的LDFALGS
LDFLAGS += $(foreach lib, $(LIBS), -l$(lib)) ##所需要的编译条件
CFLAGS += -DMERGE_MCMERGE -DMCMERGE_DEBUG -fPIC ##目标文件
TARGET := libmerge_x64.so all:$(TARGET) $(TARGET):$(FINAL_OBJS)
$(GXX) -shared -std=c++ -fPIC -m64 -g -D_DEBUG -DNDEBUG -o $@ $^ $(LDFLAGS) $(OUT_DIR)/%.o:$(SRC_DIR)/%.cpp
@test -d $(OUT_DIR) || mkdir -p $(OUT_DIR)
$(GXX) -g -c $< -o $@ $(CFLAGS) -fpermissive $(OUT_DIR)/%.o:$(SRC_DIR)/%.c
@test -d $(OUT_DIR) || mkdir -p $(OUT_DIR)
$(GCC) -g -c $< -o $@ $(CFLAGS) -fpermissive .PHONY:clean
clean:
rm -rf $(OUT_DIR)/*.o $(TARGET)

模板中涉及到的一些东西都添加了注释,需要使用的时候,只需要对地址,库名称进行更改即可,以及一些编译条件的更改。

下面对于一些这两天不太明白的地方进行注解。

1. CFLAGS(cp http://blog.chinaunix.net/uid-31387290-id-5778675.html)

使用gcc编译阶段,最常用的编译选项CFLAGS = -Wall -Werror -g -O0的解释:
-Wall:编译阶段显示所有警告。
-Werror:将所有的警告当成错误进行处理,使出现警告时就停止编译。
常见编译报错:cc1: warnings being treated as errors 。解决方法是:把-Werror去掉,不把warnning当作error处理。
-g:编译器在编译时,产生调试信息,最终产生供gdb调试使用的可执行文件。用了-g选项生成的a.out会比没用-g选项生成的a.out明显大点。
-O0:编译器的优化选项的4个级别
-O0:表示编译时没有优化。
-O1:表示编译时使用默认优化。
-O2:表示编译时使用二级优化。
-O3:表示编译时使用*优化。
-Os:相当于-O2.5优化,但又不所见代码尺寸。见:Optimization in GCC

2.  .PHONY的作用(cp https://www.cnblogs.com/idorax/p/9306528.html)

伪造的,假的。意思就是:clean是个假target,我不想生成这个target,但是我还想执行下面的语句。白嫖呗~

.PHONY:clean
clean:
rm -rf $(OUT_DIR)/*.o $(TARGET)

3. gcc和g++的区别(cp http://www.cnblogs.com/samewang/p/4774180.html)

什么是gcc / g++

首先说明:gcc 和 GCC 是两个不同的东西

GCC:GNU Compiler Collection(GUN 编译器集合),它可以编译C、C++、JAV、Fortran、Pascal、Object-C、Ada等语言。

gcc是GCC中的GUN C Compiler(C 编译器)

g++是GCC中的GUN C++ Compiler(C++编译器)

一个有趣的事实就是,就本质而言,gcc和g++并不是编译器,也不是编译器的集合,它们只是一种驱动器,根据参数中要编译的文件的类型,调用对应的GUN编译器而已,比如,用gcc编译一个c文件的话,会有以下几个步骤:

Step1:Call a preprocessor, like cpp.

Step2:Call an actual compiler, like cc or cc1.

Step3:Call an assembler, like as.

Step4:Call a linker, like ld

由于编译器是可以更换的,所以gcc不仅仅可以编译C文件

所以,更准确的说法是:gcc调用了C compiler,而g++调用了C++ compiler

gcc和g++的主要区别

①. 对于 *.c和*.cpp文件,gcc分别当做c和cpp文件编译(c和cpp的语法强度是不一样的)

②. 对于 *.c和*.cpp文件,g++则统一当做cpp文件编译

③. 使用g++编译文件时,g++会自动链接标准库STL,而gcc不会自动链接STL

④. gcc在编译C文件时,可使用的预定义宏是比较少的

⑤. gcc在编译cpp文件时/g++在编译c文件和cpp文件时(这时候gcc和g++调用的都是cpp文件的编译器),会加入一些额外的宏

⑥. 在用gcc编译c++文件时,为了能够使用STL,需要加参数 –lstdc++ ,但这并不代表 gcc –lstdc++ 和 g++等价

3. $@,$^,$<的意义
$@--目标文件,$^--所有的依赖文件,$<--第一个依赖文件。

还有各种的编译条件,有待 后续遇到问题之后,根据相关问题进行解决。

上一篇:通过实体类生成建表SQL语句实现方法


下一篇:How to Catch Ctrl-C in Shell Script