开发过程中,有这样一个需求:同一份代码,需要编译产生两种不同的输出,一种是编译成二进制可执行文件,另外一种是编译成动态链接库。考虑到代码一致性以及版本维护等因素,不希望在编译时改动目录下的任何文件名称或者后缀,因此希望通过Makefile指定某些文件可以不参与编译,具体来说,就是在编译动态链接库时,main()函数所在的cpp文件不参与编译。
实现该功能的命令为filter-out,滤除指定文件,其使用格式为:
$(filter-out<pattern...>,<text>)
以上命令的返回值为,由空格分割的“text”字串所指定的文件中所有不符合模式“PATTERN”的字串。
相反的命令有filter,filter则是选出符合pattern的文件,例如:
sources := foo.c bar.c baz.s ugh.h
newsrc = $(filter %.c %.s,$(sources))
那么newsrc为foo.c,bar.c,baz.s。
具体到本人的工程,原始Makefile文件为:
CFLAGS = `pkg-config --cflags opencv4`
LDFLAGS = `pkg-config --libs opencv4`
SOURCE = $(wildcard *.cpp)
OBJ = $(patsubst %.cpp, %.o, $(SOURCE))
EXE = demo
$(EXE):$(OBJ)
g++ -std=c++11 -o $(EXE) $^ $(LDFLAGS)
%.o:%.cpp
g++ -std=c++11 -o $@ -c $^ $(CFLAGS) -I.
.PHONY:clean
clean:
rm $(OBJ) $(EXE)
现在,我们要用这份代码编译出一份动态链接库,使用以上提到的filter-out功能,相应的Makefile为:
CFLAGS = `pkg-config --cflags opencv4` -fPIC
LDFLAGS = `pkg-config --libs opencv4` -shared
SOURCE = $(filter-out demo.cpp, $(wildcard *.cpp))
OBJ = $(patsubst %.cpp, %.o, $(SOURCE))
EXE = libdemo.so
$(EXE):$(OBJ)
g++ -std=c++11 -o $(EXE) $^ $(LDFLAGS)
%.o:%.cpp
g++ -std=c++11 -o $@ -c $^ $(CFLAGS) -I.
.PHONY:clean
clean:
rm $(OBJ) $(EXE)
其中,在SOURCE = $(filter-out demo.cpp, $(wildcard *.cpp))语句中,demo.cpp就是我们编译动态库时要滤除的源文件。
如果要排除的源文件很多,可以通过文件指定要排除的源文件列表,统一排除。类似下面的命令:
EXCLUDES := $(shell cat ./exclude.txt)
NEWSRC:= $(filter-out $(EXCLUDES),$(SRC))