通常使用GCC来编译C文件。编译过程为源文件.c文件 -> 预编译成.i文件 -> 编译成汇编语言.s -> 汇编成.o文件 -> 链接成可执行文件。编译命令为gcc -参数 .c -o 输出文件名称
gcc -E [.c源文件] -o [输出文件名.i]
gcc -S [.c源文件] -o [输出文件名.s]
gcc -c [.c源文件] [.c源文件] [...] -o [文件名]
gcc [.c源文件] [.c源文件] [...] -o [输出文件名]
gcc [.c源文件] [.c源文件] [...] -o [输出文件名] -l[库名] -L[库所在路径]
程序库是已写好的、供使用的可复用代码,每个程序都要依赖很多基础的底层库。从本质上,库是一种可执行代码的二进制形式。静态库可以在编译c项目时,将引用的库一起链接到可执行文件中,可执行文件在运行时不再需要库的支持,但可执行文件会变大。
gcc -c [.c] -o [自定义文件名]
gcc -c [.c] [.c] ...
ar -r [lib自定义库名.a] [.o] [.o] ...
gcc [.c] -o [输出文件名] -l[库名] -L[库所在路径]
动态库在程序编译时不会链接到目标代码中,而是在程序运行时才被调用,可执行文件比静态链接的可执行文件要小。不同的程序可以共享相同的动态库。
gcc -c -fpic [.c/.cpp][.c/.cpp]...
gcc -shared [.o][.o]... -o [lib自定义库名.so]
#将1和2合并为一步
gcc -fpic -shared [.c源文件] [.c源文件] [...] -o lib[库名].so
gcc [.c/.cpp] -o [自定义可执行文件名] -l[库名] -L[库路径] -Wl,-rpath=[库路径]
编译C++的流程与编译C几乎一致,只不过用的时g++命令。将上述编译.c的gcc改为g++即可编译cpp文件。静态库与动态库同理。
makefile是自动化编译的一款工具。实际中C/C++工程项目可能十分庞大,直接用手一条一条的敲gcc/g++效率极低。在makefile文件中编写编译流程可以简化编译工作。
#makefile格式
targets : prerequisties
[tab键]command
targets为生成的目标文件;
prerequisties为为了生成targets目标而所需的依赖目标文件;
conmmand为需要运行的命令
为了避免 target 和 Makefile 同级目录下 文件/文件夹 重名的这种情况,我们可以使用一个特殊的标记 .PHONY 来显式地指明一个目标是 “伪目标”,如下例中的clean
.PHONY : clean
cpp := src/main.cpp
obj := objs/main.o
cpp := src/main.cpp
obj := objs/main.o
$(obj) : ${cpp} #cpp中的字符串代表的依赖文件,obj为生成的目标文件
@g++ -c $(cpp) -o $(obj)
#等价于g++ -c src/main.cpp -o objs/main.o
compile : $(obj) #compile为伪目标,可执行make compile命令
cpp := src/main.cpp
obj := objs/main.o
$(obj) : ${cpp}
@g++ -c $< -o $@
#等价于g++ -c src/main.cpp -o objs/main.o
@echo $^
#执行shell命令echo,打印出cpp的内容
compile : $(obj)
.PHONY : compile
函数执行格式通常为$(fn, arguments) or ${fn, arguments}
# shell 指令,src 文件夹下找到 .cpp 文件
cpp_srcs := $(shell find src -name "*.cpp")
# shell 指令, 获取计算机架构
HOST_ARCH := $(shell uname -m)
cpp_srcs := $(shell find src -name "*.cpp")
cpp_objs := $(subst src/,objs/,$(cpp_objs))
cpp_srcs := $(shell find src -name "*.cpp") #shell指令,src文件夹下找到.cpp文件
cpp_objs := $(patsubst %.cpp,%.o,$(cpp_srcs)) #cpp_srcs变量下cpp文件替换成 .o文件
library_paths := /lib \
/usr/local/lib64
#每一个元素前面加上-I
I_flag := $(foreach item,$(library_paths),-I$(item))
#等价于
I_flag := $(include_paths:%=-I%)
$(dir src/foo.c hacks) # 返回值是“src/ ./”。
#嵌套使用,先用shell中的find命令寻找lib开头的文件,然后去掉这些文件的目录前缀
libs := $(notdir $(shell find /usr/lib -name lib*))
libs := $(notdir $(shell find /usr/lib -name lib*))
a_libs := $(filter %.a,$(libs)) #过滤出静态库
so_libs := $(filter %.so,$(libs)) #过滤出动态库
libs := $(notdir $(shell find /usr/lib -name lib*))
#去掉后缀和lib前缀,得到库名
a_libs := $(subst lib,,$(basename $(filter %.a,$(libs))))
so_libs := $(subst lib,,$(basename $(filter %.so,$(libs))))
objs := objs/add.o objs/minus.o objs/main.o
cpp_objs := $(filter-out objs/main.o, $(objs))
编译选项:
-m64: 指定编译为 64 位应用程序
-std=: 指定编译标准,例如:-std=c++11、-std=c++14
-g: 包含调试信息
-w: 不显示警告
-O: 优化等级,通常使用:-O3
-I: 加在头文件路径前
fPIC: (Position-Independent Code), 产生的没有绝对地址,全部使用相对地址,代码可以被加载到内存的任意位置,且可以正确的执行。这正是共享库所要求的,共享库被加载时,在内存的位置不是固定的
链接选项:
-l: 加在库名前面
-L: 加在库路径前面
-Wl,<选项>: 将逗号分隔的 <选项> 传递给链接器
-rpath=: “运行” 的时候,去找的目录。运行的时候,要找 .so 文件,会从这个选项里指定的地方去找
cpp_srcs := $(shell find src -name *.cpp)
cpp_objs := $(patsubst src/%.cpp,objs/%.o,$(cpp_srcs))
include_dirs :=/home/include
I_options := $(include_dirs:%=-I%)
compile_options := -g -O3 -w $(I_options)
lib_dirs := /home/lib #库路径
link_libs := xxx #库名
L_options := $(lib_dirs:%=-L%)
l_options := $(link_libs:%=-l%)
link_options := $(l_options) $(L_options) $(I_options)
#每个.cpp生成一个对应的目标文件
objs/%.o : src/%.cpp
#创建文件夹,@表示make的时候不打印make信息
@mkdir -p $(dir $@)
@g++ -c $^ -o $@ $(compile_options)
workspace/exec : $(cpp_objs)
@mkdir workspace/exec
@g++ $^ -o $@ $(link_options)
# 输入make run即可生成run伪目标
run : workspace
@./$<
# 输入make clean即可生成clean伪目标
clean :
@rm -rf objs workspace/exec
debug :
@echo $(as_files)
.PHONY : debug run clean