这是我自己写的一份makefile文件,最简单的一个
BSoftInst: BSoftInst.cpp ReadXML.cpp main.cpp ireg.cpp ntreg.cpp SoftwareInst.cpp
g++ -o BSoftInst BSoftInst.cpp ReadXML.cpp main.cpp ireg.cpp ntreg.cpp SoftwareInst.cpp -Wformat -fpermissive -I/usr/include/libxml2 -L/usr/lib/i386-linux-gnu -lxml2 -lguestfs
1.makefile的书写规则为
target ... : prerequisites ...
[Tab] command
target是需要要生成的目标文件
prerequisites是生成目标所依赖的文件
command是命令,其开头需要是一个tab,而不是一系列空格
2.变量的使用
关于编译程序,和编译选项,我们是可以使用变量指定的,这样方便维护,特别当你的工程很大的时候
如可以改成下面这个形式
#Which compiler
CC = g++
#Where are include files kept
INCLUDE = .
#Options for development
CFLAGS = -g -Wall -ansi -Wformat -fpermissive -I/usr/include/libxml2 -L/usr/lib/i386-linux-gnu -lxml2 -lguestfs
#Options for release
#CFLAGS = -O -Wall -ansi
BSoftInst: BSoftInst.cpp ReadXML.cpp main.cpp ireg.cpp ntreg.cpp SoftwareInst.cpp
$(CC) -o BSoftInst BSoftInst.cpp ReadXML.cpp main.cpp ireg.cpp ntreg.cpp SoftwareInst.cpp
3.完整的依赖关系
之前的版本,是把所有.cpp文件都罗列在一块了,没有清晰的罗列出来他们之间的依赖关系。
我是为了偷懒,但是当文件很多是,这种方式就不行了。需要想下面那样罗列他们之间的关系
BSoftInst: BSoftInst.o ReadXML.o main.o ireg.o ntreg.o SoftwareInst.o
$(CC) -o $@ BSoftInst.o ReadXML.o main.o ireg.o ntreg.o SoftwareInst.o
BSoftInst.o: BSoftInst.cpp BSoftInst.h ReadXML.h
ReadXML.o: ReadXML.cpp ReadXML.h
main.o: main.cpp ireg.h BSoftInst.h ReadXML.h
ireg.o: ireg.cpp ireg.h ntreg.h
ntreg.o: ntreg.cpp ntreg.h
SoftwareInst.o: SoftwareInst.cpp SoftwareInst.h
也可以在依赖关系的后面加上编译命令,如
BSoftInst.o: BSoftInst.cpp BSoftInst.h ReadXML.h
g++ -c BSoftInst.cpp BSoftInst.h ReadXML.h
4.伪目标的使用
这个编译过程会产生很多临时文件,如大量的.o文件,需要删除,我们可以在makefile文件中
这样写
clean:
rm -f *.o
这两行也符合makefile的规则(
target+
prerequisites+
command),只不过其中的依赖关系(
prerequisites
)没有了。clean被称为伪目标。
“伪目标”并不是一个文件,只是一个标签,
由于“伪目标”不是文件,所以make无法生成它的依赖关系和决定它是否要执行。我们只有
通过显示地指明这个“目标”才能让其生效。
在命令行输入make clean 后就会执行删除操作了。
当然,“伪目标”的取名不能和文件名重名,为了确保不重名,需要这样做
.PHONY : clean
只要有这个声明,
不管是否有
“clean”
文件,
要运行
“clean”
这个目标, “make clean”就可以执行之后的命令。
5.默认目标
Makefile中的第一个目标会被作为其默认目标,我们会声明一个伪目标在文件开头,像这样
all : BSoftInst
这样输入make,就相当于执行make BSoftInst。当你的makefile文件中有好多个目标的时候,会很有用。如果你想输入make一口气编译多个目标需要这样做
all : prog1 prog2 prog3
.PHONY : all
prog1 : prog1.o utils.o
cc -o prog1 prog1.o utils.o
prog2 : prog2.o
cc -o prog2 prog2.o
prog3 : prog3.o sort.o utils.o
cc -o prog3 prog3.o sort.o utils.o
6.对于makefile,我也就知道这么多了。
cmake的简单使用
1.简单使用
kdevelop4目前就使用的是cmake,cmake需要CMakeLists.txt
下面是我的kdevelop生成的一个CMakeLists.txt文件,看上去写一个
CMakeLists.txt要比写makefile简单的多。
project(analysisreg2)
add_executable(analysisreg2 main.cpp ntreg.cpp edlib.cpp)
PROJECT 指令的语法是:
PROJECT(projectname [CXX] [C] [Java])
你可以用这个指令定义工程名称,并可指定工程支持的语言,支持的语言列表是可以忽略的,
默认情况表示支持所有语言。
ADD_EXECUTABLE(analysisreg2 main.cpp ntreg.cpp edlib.cpp)
定义了这个工程会生成一个文件名为 hello 的可执行文件,后面的参数是其依赖的源文件
cmake会生成一个makefile,而且会针对不同平台生成不同的makefile。
2.变量的使用
PROJECT (Analysisreg2)
SET(SRC_LIST main.cpp ntreg.cpp edlib.cpp)
MESSAGE(STATUS "This is BINARY dir " ${Analysisreg2_BINARY_DIR})
MESSAGE(STATUS "This is SOURCE dir "${Analysisreg2_SOURCE_DIR})
ADD_EXECUTABLE(analysisreg2 SRC_LIST)
SET 指令的语法是:
SET(VAR [VALUE] [CACHE TYPE DOCSTRING [FORCE]])
现阶段,你只需要了解 SET 指令可以用来显式的定义变量即可。
比如我们用到的是 SET(SRC_LIST
main.c),如果有多个源文件,也可以定义成:
SET(SRC_LIST main.c t1.c t2.c)
MESSAGE 指令的语法是:
MESSAGE([SEND_ERROR | STATUS | FATAL_ERROR] "message to display"
...)
这个指令用于向终端输出用户定义的信息,包含了三种类型:
SEND_ERROR,产生错误,生成过程被跳过。
SATUS,输出前缀为—的信息。
FATAL_ERROR,立即终止所有 cmake 过程.
我们在这里使用的是 STATUS 信息输出,演示了由 PROJECT 指令定义的两个隐式变量
HELLO_BINARY_DIR 和 HELLO_SOURCE_DIR。