Linux与Windows下静态库(.a/.lib)编译时合并代码、无运行时依赖,动态库(.so/.dll)运行时加载、支持共享;均需头文件声明,动态库须控制符号导出,C接口更跨平台。
在 Linux 和 Windows 上创建和使用静态库与动态库,核心区别在于编译链接方式和运行时依赖机制,但基本流程相似:先编译源码为目标文件,再归档或打包为库,最后在主程序中链接调用。
静态库是目标文件(.o / .obj)的集合,链接时直接复制代码到可执行文件中,运行时不需额外库文件。
g++ -c math.cpp -o math.o 编译为目标文件;ar rcs libmath.a math.o 打包成静态库;g++ main.cpp -L. -lmath -o main(-L 指定路径,-l 自动找 libxxx.a)。cl /c math.cpp 生成 math.obj;lib math.obj /OUT:math.lib 创建静态库;cl main.cpp math.lib 或在项目属性 → “附加依赖项”中添加 math.lib。动态库在程序运行时才载入内存,多个程序可共用同一份库文件,便于热更新和节省内存。
-fPIC 生成位置无关代码:g++ -fPIC -c math.cpp -o math.o;g++ -shared -o libmath.so math.o 生成共享库;g++ main.cpp -L. -lmath -o main,运行前需确保 libmath.so 在 LD_LIBRARY_PATH 中或系统路径下;,适用于插件式架构。__declspec(dllexport) 标记导出函数(如 extern "C" __declspec(dllexport) int add(int a, int b););cl /EHsc /LD math.cpp(/LD 生成 DLL),会同时输出 math.dll 和 math.lib(导入库);math.lib(用于编译期解析符号),运行时需保证 math.dll 在 PATH 或可执行目录中;
LoadLibrary + GetProcAddress 动态加载,不依赖导入库。无论静态还是动态,调用方都需要头文件声明接口。动态库还需注意符号导出控制:
__attribute__((visibility("hidden"))) 隐藏内部符号,仅对需要导出的函数加 __attribute__((visibility("default")));__declspec(dllexport)(构建 DLL 时)和 __declspec(dllimport)(使用 DLL 时),常用宏封装避免重复判断,例如:#ifdef BUILDING_MATH_DLL
#define MATH_API __declspec(dllexport)
#else
#define MATH_API __declspec(dllimport)
#endif
MATH_API int add(int, int);
纯 C 接口最易跨平台;C++ 类、模板、异常、RTTI 在动态库边界易出问题,建议只导出 C 风格函数。
std::string、std::vector),不同编译器/标准库实现可能不兼容;create_xxx()/destroy_xxx());add_library(math STATIC ...) 或 add_library(math SHARED ...),自动处理标志和后缀。