编译过程
1.预处理 g++ –E hello.c –o hello.i //-o 指定输出名称
2.编译 g++ –S hello.i –o hello.s //.i可换成.c直接编译 c语言转换成汇编语言
3.汇编 g++ –c hello.s –o hello.o //.s可换成.c 汇编转机器码 可跳过前两步直接用-c
4.链接 g++ hello.o –o hello [--static] //.o可换成.c一步生成
默认动态链接,运行时加载外部代码库;
--static 静态链接 把外部库打包到自己的可执行文件中(不依赖外部环境)
--shared 动态链接
-g 只是编译器,在编译的时候,产生调试信息
-I:添加包含路径path,表示将后面的目录加入g++头文件的搜索路径(在程序中#include时) //先搜path,再从默认/usr/include-->/usr/local/include
linux下的库文件都应有“lib”前缀
创建静态库
先-c编译成.o目标文件,linux下用 ar 命令生成静态库
ar crv libmyAPI.a myAPI.o //生成libmyAPI.a静态库
g++ main.cpp libmyAPI.a -o output //使用静态库生成output可执行文件,库文件作为依赖项要放到后面
创建动态库
g++ one.cpp two.cpp three.cpp -fPIC -shared -o libtest.so //生成动态库,必须动态链接
g++ main.cpp -L. -ltest -o main //使用动态库
-fPIC 表示编译为位置独立的代码
-ltest 表示使用名为libtest.so的动态库
-L. 表示要连接的库文件在当前目录中
ldd test 可查看test是否链接到动态库
若设置了-I 和-L.还是未链上,则可修改环境变量LD_LIBRARY_PATH
LD_LIBRARY_PATH:这个环境变量指示动态连接器可以装载动态库的路径 指定动态库目录
ar 常用命令:
ar rc/rcv archive files... 新建库
ar r/rs archive files... 在已存在的库插入
ar 命令格式:
ar [-]{dmpqrtx}[abcfilNoPsSuvV] [membername] [count] archive files...
操作选项{dmpqrtx}(在命令中只能并且必须使用其中一个):
d:从库中删除模块。按模块原来的文件名指定要删除的模块。如果使用了任选项v则列出被删除的每个模块。
m:该操作是在一个库中移动成员。当库中如果有若干模块有相同的符号定义(如函数定义),则成员的位置顺序很重要。如果没有指定任选项,任何指定的成员将移到库的最后。也可以使用'a','b',或'i'任选项移动到指定的位置。
p:显示库中指定的成员到标准输出。如果指定任选项v,则在输出成员的内容前,将显示成员的名字。如果没有指定成员的名字,所有库中的文件将显示出来。
q:快速追加。增加新模块到库的结尾处。并不检查是否需要替换。'a','b',或'i'任选项对此操作没有影响,模块总是追加的库的结尾处。如果使用了任选项v则列出每个模块。 这时,库的符号表没有更新,可以用'ar s'或ranlib来更新库的符号表索引。
r:在库中插入模块(替换)。当插入的模块名已经在库中存在,则替换同名的模块。如果若干模块中有一个模块在库中不存在,ar显示一个错误消息,并不替换其他同名模块。默认的情况下,新的成员增加在库的结尾处,可以使用其他任选项来改变增加的位置。
t:显示库的模块表清单。一般只显示模块名。
x:从库中提取一个成员。如果不指定要提取的模块,则提取库中所有的模块。
选项参数:
a:在库的一个已经存在的成员后面增加一个新的文件。如果使用任选项a,则应该为命令行中membername参数指定一个已经存在的成员名。
b:在库的一个已经存在的成员前面增加一个新的文件。如果使用任选项b,则应该为命令行中membername参数指定一个已经存在的成员名。
c:创建一个库。不管库是否存在,都将创建。
f:在库中截短指定的名字。缺省情况下,文件名的长度是不受限制的,可以使用此参数将文件名截短,以保证与其它系统的兼容。
i:在库的一个已经存在的成员前面增加一个新的文件。如果使用任选项i,则应该为命令行中membername参数指定一个已经存在的成员名(类似任选项b)。
l:暂未使用
N:与count参数一起使用,在库中有多个相同的文件名时指定提取或输出的个数。
o:当提取成员时,保留成员的原始数据。如果不指定该任选项,则提取出的模块的时间将标为提取出的时间。
P:进行文件名匹配时使用全路径名。ar在创建库时不能使用全路径名(这样的库文件不符合POSIX标准),但是有些工具可以。
s:写入一个目标文件索引到库中,或者更新一个存在的目标文件索引。甚至对于没有任何变化的库也作该动作。对一个库做ar s等同于对该库做ranlib。
S:不创建目标文件索引,这在创建较大的库时能加快时间。
u:一般说来,命令ar r...插入所有列出的文件到库中,如果你只想插入列出文件中那些比库中同名文件新的文件,就可以使用该任选项。该任选项只用于r操作选项。
v:该选项用来显示执行操作选项的附加信息。
V:显示ar的版本。
编译报错 [Error] ld returned 1 exit status
1.程序没有主函数
2.printf/scanf书写错误
3.1在主函数中出现自定义函数名书写错误(即出现在主函数前未出现的自定义函数)
3.2当自定义函数只进行了声明,主函数便进行调用
4.上一个运行结果未关闭
其他
.cpp 用g++
.c 用gcc
手动编译gcc-9.3.0 大概3-4小时
手动升级gcc不会更新老版本的动态库,见https://blog.csdn.net/feikudai8460/article/details/113699655
创建库文件参考地址
在win下使用gcc,安装MinGW
重新打开cmd,新加的环境变量才会生效
g++编译的.hpp无法执行