目录
C/C++库文件全了解
-
一. 前言
是不是老是在编译的过程出现各种奇奇怪怪的错误,对于出现的错误完全看不懂 在说点啥,只会复制黏贴到百度一通瞎搜索?是不是对于一些库文件.dll,.so,.lib完全不知道是个啥,完全不知道怎么使用他们?这是因为我们对一个C/C++程序的编译过程不够了解。关于编译过程的概念我放到了我的思维导图,链接,这里不再赘述。本文旨在带你了解库文件的编译生成和调用。
-
二. 名词解释
-
2.1. windows的静态库:xxx.lib
包含testlib.lib和testlib.h两个文件,在写程序调用静态库里包含的函数的时候,编译完就已经把函数的实现放到了产生的exe文件里了,所以exe后期的运行不需要依赖静态库了,代价就是exe体积会变大。目前主流程序一般不用静态库 -
2.2. windows的动态库:xxx.dll
包含testlib.lib和testlib.dll两个文件,在写程序调用动态库里包含的函数的时候,编译阶段只把函数的重定位信息放到了exe文件里了,所以exe后期的运行需要依赖动态库。值得一提的是,虽然动态库和静态库都有一个.lib文件,但是两者完全不一样。动态库的.lib文件只在你调用的编译阶段需要,而.dll只在调用的运行阶段需要。 -
2.3. linux的动态库:libxxx.so
概念和windows的动态库一样,就是linux下的静态库和动态库命名时都需要在前面加lib -
2.4. linux的静态库:libxxx.a
-
-
三. 自己编译库
-
3.1 编译产生windows的静态库
-
3.1.1代码准备:
C:\Users\john\Desktop\static_lib:
├─static_lib.cpp
├─static_lib.h
├─pch.cpp #这两个是vc++自己产生的的预编译头,这里我们不关注
└─pch.h #这两个是vc++自己产生的的预编译头,这里我们不关注 -
3.1.2 static_lib.cpp:
#include "pch.h" #include "static_lib.h" int add(int x, int y) { return x + y; }
-
3.1.3 static_lib.h:
#pragma once int add(int x, int y);
编译完成后得到DEBUG文件夹里得到static_lib.lib文件,另外加一个刚才用的static_lib.h,有这两个文件就可以调用函数add了,调用过程在后面。
-
-
3.2 编译产生windows的动态库
-
3.2.1代码准备:
C:\Users\john\Desktop\dynamic_lib:
├─dynamic_lib.cpp
├─dynamic_lib.h
├─pch.cpp #这两个是vc++自己产生的的预编译头,这里我们不关注
└─pch.h #这两个是vc++自己产生的的预编译头,这里我们不关注 -
3.2.2 dynamic_lib.cpp:
#include "pch.h" #include "dynamic_lib.h" #define DLL_API _declspec(dllexport) DLL_API int add(int a, int b) //实现两个整数相加 { return a + b; }
-
3.2.3 dynamic_lib.h:
#pragma once #define DLL_API _declspec(dllexport) int add(int x, int y);
编译完在DEBUG里会产生dynamic_lib.dll和dynamic_lib.lib两个文件
-
-
3.3 编译产生linux的静态库
- 3.3.1代码准备:
/home/nvidia/DeepLearning/test_a:
├─static_lib.c
└─static_lib.h - 3.3.2 static_lib.c:
#include "static_lib.h" int add(int a, int b) { int c; c = a + b; return c ; }
- 3.3.3 static_lib.h:
#pragma onece int add(int a, int b);
- 3.3.4 编译生成静态库:
编译过程分两步,先生成.o文件,然后根据.o文件归档为.a静态库文件
然后归档为静态库文件,这样就生成了静态库libstatic_lib.agcc -c static_lib.c
ar crv libstatic_lib.a static_lib.o
- 3.3.1代码准备:
-
3.4 编译产生linux的动态库
-
3.4.1代码准备:
/home/nvidia/DeepLearning/test_so:
├─dynamic_lib.c
└─dynamic_lib.h -
3.4.2 dynamic_lib.c:
#include "dynamic_lib.h" int add(int a, int b) { int c; c = a + b; return c ; }
-
3.4.3 dynamic_lib.h:
#pragma onece int add(int a, int b);
编译产生.so文件,命令行里输入
nvidia@tegra-ubuntu:~$ gcc dynamic_lib.c -shared -fPIC -o libdynamic_lib.so
要注意的是,一般产生的.so文件名要以lib开头
-
-
-
四. 自己调用库
-
4.1. 调用windows静态库文件
- 4.1.1代码准备:
C:\Users\john\Desktop\lib_test:
├─lib_test.cpp
└─stdafx.h #这是vc++自己产生的的预编译头,这里我们不关注 - 4.1.2 lib_test.cpp:
#include "stdafx.h" #include "static_lib.h" #pragma comment(lib, "static_lib.lib") int _tmain(int argc, _TCHAR* argv[]) { int a = 0, b = 2, c; c = add(a, b); return 0; }
- 4.1.1代码准备:
-
4.2. 调用windows动态库文件
- 4.2.1代码准备:
C:\Users\john\Desktop\lib_test:
├─lib_test.cpp
└─stdafx.h #这是vc++自己产生的的预编译头,这里我们不关注 - 4.2.2 lib_test.cpp:
#include "stdafx.h" #pragma comment(lib, "dynamic_lib.lib") extern int add(int a, int b); int _tmain(int argc, _TCHAR* argv[]) { int a = 0, b = 2, c; c = add(a, b); return 0; }
- 4.2.1代码准备:
-
4.3. 调用linux静态库文件
- 4.3.1代码准备:
/home/nvidia/DeepLearning/lib_test:
└─lib_test.c - 4.3.2 lib_test.c:
#include "static_lib.h" int main() { int a = 0, b = 2, c; c = add(a, b); return 0; }
- 4.3.3 编译:
gcc lib_test.c -o lib_test -L ../static_lib/ -lstatic_lib -I ../static_lib/
- 4.3.4 运行即可成功
- 4.3.1代码准备:
-
4.4. 调用linux动态库文件
- 4.4.1代码准备:
/home/nvidia/DeepLearning/lib_test:
└─lib_test.c - 4.4.2 lib_test.c:
#include "dynamic_lib.h" int main() { int a = 0, b = 2, c; c = add(a, b); return 0; }
- 4.4.3 编译:
gcc lib_test.c -o lib_test -L ../dynamic_lib/ -ldynamic_lib -I ../dynamic_lib/
- 4.4.3 运行报错:
直接输入./lib_test
运行会报错。因为刚才编译的时候可以通过编译参数告诉编译器动态库的位置,但是运行的时候不知道动态库在哪。
通过ldd命令可以查看lib_test需要哪些动态库,可以看到我们的动态库他找不到./lib_test: error while loading shared libraries: libdynamic_lib.so: cannot open shared object file: No such file or directory
bash huangshiqing@ezviz-W580-G20:~/lib_test$ ldd lib_test linux-vdso.so.1 => (0x00007ffd659e0000) libdynamic_lib.so => not found libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f0a01391000) /lib64/ld-linux-x86-64.so.2 (0x00007f0a0175b000)
- 4.4.4 指定动态库搜索路径:
动态链接时、执行时搜索路径顺序:
- 4.4.1代码准备:
- 编译目标代码时指定的动态库搜索路径;
只在编译环节可以通过命令参数指定 - 环境变量LD_LIBRARY_PATH指定的动态库搜索路径;
通过下面命令临时添加或者在.bashrc中永久添加export LD_LIBRARY_PATH=/home/nvidia/DeepLearning/dynamic_lib/:$LD_LIBRARY_PATH
- 配置文件/etc/ld.so.conf中指定的动态库搜索路径;
概念参见这里 - 默认的动态库搜索路径/lib;
直接将动态库移到这两个默认会去搜索的路径也可以 - 默认的动态库搜索路径/usr/lib。
- 4.4.5 指定动态库位置后再次运行即成功
参考链接:https://blog.youkuaiyun.com/sunxiaopengsun/article/details/79351025
-