最近因为编译需求,需要更改一些编译条件,顺带看了一些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++11 -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. $@,$^,$<的意义
$@--目标文件,$^--所有的依赖文件,$<--第一个依赖文件。
还有各种的编译条件,有待 后续遇到问题之后,根据相关问题进行解决。