c++怎么编译生成一个动态链接库dll或so_c++共享库制作与导出方法

在C++中生成动态链接库需根据平台使用不同导出方式:Windows下用__declspec(dllexport)配合/MD或/LD编译,Linux下用-fPIC和-shared生成SO,通过预定义宏统一接口,结合extern "C"避免名称修饰,确保符号正确导出。

在C++中编译生成动态链接库(Windows下为DLL,Linux下为SO)需要根据平台使用不同的编译选项和符号导出方式。核心在于正确声明要导出的函数或类,并用编译器支持的方式生成共享库文件。

Windows下生成DLL

在Windows上使用MSVC或MinGW编译器可以生成DLL文件。关键是要使用__declspec(dllexport)导出符号。

示例代码(mathlib.h):

#ifndef MATHLIB_H
#define MATHLIB_H

#ifdef MATHLIB_EXPORTS
    #define MATHLIB_API __declspec(dllexport)
#else
    #define MATHLIB_API __declspec(dllimport)
#endif

extern "C" MATHLIB_API int add(int a, int b);
class MATHLIB_API MathClass {
public:
    int multiply(int a, int b);
};

#endif

实现文件(mathlib.cpp):

#define MATHLIB_EXPORTS
#include "mathlib.h"

extern "C" MATHLIB_API int add(int a, int b) {
    return a + b;
}

int MathClass::multiply(int a, int b) {
    return a * b;
}

使用Visual Studio命令行编译DLL:

cl /LD mathlib.cpp /link /out:mathlib.dll

说明:
- /LD 表示生成DLL
- 定义 MATHLIB_EXPORTS 使头文件中的宏展开为 dllexport
- 使用 extern "C" 可避免C++名称修饰,便于外部调用

Linux下生成SO(共享库)

Linux默认导出所有全局符号,但推荐使用可见性属性控制导出,提升性能并减少体积。

修改头文件(mathlib.h):

#ifndef MATHLIB_H
#define MATHLIB_H

#ifdef __cplusplus
extern "C" {
#endif

#ifdef _WIN32
    #ifdef MATHLIB_EXPORTS
        #define MATHLIB_API __declspec(dllexport)
    #else
        #define MATHLIB_API __declspec(dllimport)
    #endif
#else
    #if defined(__GNUC__) && (__GNUC__ >= 4)
        #define MATHLIB_API __attribute__((visibility("default")))
    #else
        #define MATHLIB_API
    #endif
#endif

MATHLIB_API int add(int a, int b);

#ifdef __cplusplus
}
#endif

#endif

编译生成SO:

g++ -fPIC -shared mathlib.cpp -o libmathlib.so

说明:
- -fPIC:生成位置无关代码,必要选项
- -shared:指示生成共享库
- 使用 visibility("default") 显式导出符号,配合编译选项 -fvisibility=hidden 可隐藏未标记符号

跨平台兼容处理

通过预处理器定义统一接口,使代码可在多个平台使用。

编译时导出定义:

  • Windows (MSVC/MinGW): 编译库时加 D MATHLIB_EXPORTS
  • Linux: 无需额外定义,除非使用 visibility 控制

例如Linux编译命令增强版:

g++ -fPIC -fvisibility=hidden -shared mathlib.cpp -o libmathlib.so

这样只有标记 MATHLIB_API 的函数才会被导出。

验证导出符号

检查生成的库是否包含期望的符号:

  • Windows: 使用 dumpbin /exports mathlib.dll
  • Linux: 使用 nm -D libmathlib.soreadelf -s libmathlib.so

确保函数名未被过度修饰(特别是C函数建议用 extern "C" 包裹)。

基本上就这些。关键是根据不同平台设置正确的导出宏和编译选项,保持接口清晰稳定。跨平台项目可结合 CMake 管理构建流程,自动处理差异。