0

0

c++如何编写动态链接库(DLL/so)_c++共享库的创建与使用【教程】

尼克

尼克

发布时间:2025-12-25 12:11:02

|

259人浏览过

|

来源于php中文网

原创

C++动态库需跨平台导出符号:Windows用__declspec(dllexport/import),Linux/macOS用__attribute__((visibility("default")))配合-fvisibility=hidden;编译时Windows用cl /LD,Linux用g++ -shared -fPIC,macOS用clang++ -dynamiclib -fPIC;调用支持静态链接或动态加载。

c++如何编写动态链接库(dll/so)_c++共享库的创建与使用【教程】

在 C++ 中编写动态链接库(Windows 下叫 DLL,Linux/macOS 下叫 shared object,即 .so.dylib),核心是导出函数/类供外部调用,同时注意平台差异和编译链接方式。下面分步骤讲清楚怎么创建、编译、使用。

一、编写可导出的 C++ 代码

动态库不是直接运行的程序,而是提供功能的“工具箱”。关键是要明确哪些符号(函数、类)需要被外部看到。

  • Windows(DLL):用 __declspec(dllexport) 标记要导出的函数或类;用 __declspec(dllimport) 在调用端声明(通常用宏自动切换)
  • Linux/macOS(.so):默认隐藏所有符号,用 __attribute__((visibility("default"))) 显式标记要导出的符号(推荐开启 -fvisibility=hidden 编译选项提升安全性)

示例(跨平台写法,头文件 math_utils.h):

#ifdef __cplusplus
extern "C" {
#endif

ifdef _WIN32

ifdef MATH_UTILS_EXPORTS

#define MATH_API __declspec(dllexport)

else

#define MATH_API __declspec(dllimport)

endif

else

define MATH_API attribute((visibility("default")))

endif

MATH_API int add(int a, int b); MATH_API int multiply(int a, int b);

ifdef __cplusplus

}

立即学习C++免费学习笔记(深入)”;

羚珑
羚珑

京东推出的一站式AI图像处理平台

下载

endif

实现文件 math_utils.cpp 只需普通实现,不加额外修饰:

#include "math_utils.h"
int add(int a, int b) { return a + b; }
int multiply(int a, int b) { return a * b; }

二、编译生成动态库

命令依赖编译器和平台,但逻辑一致:告诉编译器“这不是可执行程序,是共享库”,并处理符号可见性。

  • Windows(MSVC,命令行):
    cl /LD /Fe:math_utils.dll math_utils.cpp
    /LD 表示生成 DLL,/Fe: 指定输出名)
  • Linux(g++/clang):
    g++ -fPIC -shared -fvisibility=hidden -o libmath_utils.so math_utils.cpp
    -fPIC 生成位置无关码,-shared 生成 so,-fvisibility=hidden 配合头文件中的 visibility("default") 控制导出)
  • macOS(clang):
    clang++ -dynamiclib -fPIC -fvisibility=hidden -o libmath_utils.dylib math_utils.cpp

三、在主程序中调用动态库

分两种方式:静态链接(编译时绑定)、动态加载(运行时加载)。前者简单常用,后者更灵活(比如插件系统)。

  • 静态链接方式(推荐初学)
    写主程序 main.cpp,包含头文件并直接调用函数:
    #include "math_utils.h"
    #include 
    int main() {
        std::cout << add(3, 4) << "\n";        // 输出 7
        std::cout << multiply(3, 4) << "\n";    // 输出 12
    }
    编译时链接库:
    Windows:cl main.cpp /link math_utils.lib
    Linux:g++ main.cpp -L. -lmath_utils -o main(假设 libmath_utils.so 在当前目录)
    macOS:clang++ main.cpp -L. -lmath_utils -o main
  • 动态加载方式(跨平台需封装)
    Windows 用 LoadLibrary + GetProcAddress
    Linux/macOS 用 dlopen + dlsym
    优点是无需编译时依赖库文件,支持热插拔;缺点是手动管理符号、类型安全弱、易出错。

四、注意事项与常见问题

  • 避免导出 C++ 类的完整定义(尤其含 STL 成员)——不同编译器/标准库 ABI 不兼容。建议只导出 C 风格函数,或用 PIMPL + 工厂函数封装类
  • Windows DLL 要确保运行时一致(如都用 MT 或 MD 链接 CRT),否则可能崩溃
  • Linux 下运行时报 “xxx: cannot open shared object file”:检查 LD_LIBRARY_PATH 是否包含库路径,或用 ldd ./main 查依赖
  • 导出 C++ 函数名会被编译器修饰(mangling),所以 C 接口必须用 extern "C" 包裹,否则调用端找不到符号

基本上就这些。写动态库不复杂,关键是理解“导出控制”和“链接时机”两个核心点。从写头文件开始,统一宏定义,再按平台编译,最后链接调用——流程清晰,一次搞定。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

311

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

510

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

46

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

177

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

980

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

36

2025.10.17

default gateway怎么配置
default gateway怎么配置

配置default gateway的步骤:1、了解网络环境;2、获取路由器IP地址;3、登录路由器管理界面;4、找到并配置WAN口设置;5、配置默认网关;6、保存设置并退出;7、检查网络连接是否正常。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

218

2023.12.07

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

487

2023.07.26

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 5.9万人学习

Git 教程
Git 教程

共21课时 | 2.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号