0

0

C++怎么写动态库 C++生成DLL与调用DLL详细步骤【工程】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-02 18:41:02

|

475人浏览过

|

来源于php中文网

原创

正确配置生成DLL需确保三点:项目属性设为Dynamic Library;源文件用__declspec(dllexport)或.def文件导出符号;头文件用宏区分dllimport/dllexport并加extern "C"防名字修饰。

c++怎么写动态库 c++生成dll与调用dll详细步骤【工程】

Visual Studio 项目中怎么正确配置生成 DLL

生成 DLL 的关键不是写代码,而是让编译器知道“这个工程要导出符号”。默认新建的 C++ 动态库项目(DLL)已设好基础配置,但手动建空项目或从静态库改过来时极易漏掉 __declspec(dllexport) 或项目属性设置。

必须确认以下三点:

  • 项目属性 → 配置属性 → 常规 → “配置类型”为 Dynamic Library (.dll)
  • C++ 源文件中,所有要被外部调用的函数/类,必须显式标记导出:用 __declspec(dllexport)(Windows 特有),或更推荐的方式——通过模块定义文件(.def)统一导出
  • 若使用 __declspec(dllexport),建议配合宏封装,避免在调用方误用:例如定义 #define MYDLL_API __declspec(dllexport),并在头文件中对导出函数加该宏;调用方则用 __declspec(dllimport)

不加导出声明的函数,即使编译成 DLL,链接时也会报 LNK2019: unresolved external symbol

头文件里如何安全声明 DLL 导出函数

头文件是调用方和实现方的契约。写错会导致编译通过但运行崩溃,或链接失败。

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

典型错误是直接在头文件里写 __declspec(dllexport) —— 这会让调用方也尝试导出,引发重复定义或链接错误。

正确做法是用宏区分编译角色:

#ifdef BUILDING_MYDLL
    #define MYDLL_API __declspec(dllexport)
#else
    #define MYDLL_API __declspec(dllimport)
#endif

extern "C" { MYDLL_API int add(int a, int b); }

其中 BUILDING_MYDLL 是你在 DLL 工程的“预处理器定义”里手动添加的宏(项目属性 → C/C++ → 预处理器 → 预处理器定义)。这样头文件在 DLL 编译时展开为 dllexport,在调用方编译时展开为 dllimport

extern "C" 是为了禁用 C++ 名字修饰(name mangling),否则 C++ 函数导出后名字乱码,C# 或其他语言根本没法调用。

萝卜简历
萝卜简历

免费在线AI简历制作工具,帮助求职者轻松完成简历制作。

下载

隐式链接调用 DLL 的完整步骤(.lib + .h + .dll 缺一不可)

这是最常用、最稳妥的调用方式,但三者路径和内容必须严格匹配,否则运行时报 0xc000007b找不到指定模块

调用方工程需配置:

  • 包含目录:指向 DLL 工程的头文件所在路径(如 $(SolutionDir)MyDll\include
  • 库目录:指向生成的 .lib 文件所在目录(如 $(OutDir),即 DLL 输出目录)
  • 附加依赖项:填入 MyDll.lib(注意不是 MyDll.dll
  • 运行时把 MyDll.dll 放到调用方可执行文件同目录下(或系统 PATH 路径中)

常见坑:

  • 用了 x64 DLL,但调用方是 Win32 配置 → 直接加载失败,错误码 ERROR_BAD_EXE_FORMAT
  • 忘记复制 .dll 到输出目录,调试时提示 LoadLibrary failed: 126
  • 头文件里声明了函数,但 DLL 工程没实际实现,或导出名不一致(尤其没加 extern "C" 时)→ 运行时 GetProcAddress 返回 NULL

用 LoadLibrary + GetProcAddress 手动加载 DLL(适用于插件场景)

这种方式绕过链接期绑定,适合运行时决定是否加载、或多版本共存等场景,但写法稍繁琐,且失去编译期类型检查。

核心是三步:

  • LoadLibrary(L"MyDll.dll") 获取模块句柄(返回 HMODULE
  • GetProcAddress(hModule, "add") 获取函数地址(注意:C++ 函数名需用修饰后名字,除非加了 extern "C";C 函数名就是字符串字面量)
  • 把返回值强制转为函数指针类型再调用,例如:typedef int (*ADD_FUNC)(int, int); ADD_FUNC pAdd = (ADD_FUNC)GetProcAddress(...); int r = pAdd(3, 4);

必须检查每一步返回值:如果 LoadLibrary 返回 NULL,用 GetLastError() 查具体原因(比如路径错、依赖缺失、位数不匹配);如果 GetProcAddress 返回 NULL,大概率是函数名拼错或未导出。

别忘了最后调用 FreeLibrary(hModule),否则 DLL 句柄泄漏,多次加载后进程无法退出。

导出类比导出函数更复杂,涉及虚表、内存分配器一致性等问题,除非明确需要跨 DLL 边界传递对象,否则优先用纯 C 风格函数接口。另外,DLL 中尽量避免使用 STL 容器作为参数或返回值——不同模块可能用不同版本 CRT,导致析构崩溃。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

238

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

499

2024.03.01

typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

109

2023.09.26

define的用法
define的用法

define用法:1、定义常量;2、定义函数宏:3、定义条件编译;4、定义多行宏。更多关于define的用法的内容,大家可以阅读本专题下的文章。

340

2023.10.11

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

361

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1505

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

625

2023.11.24

AO3官网入口与中文阅读设置 AO3网页版使用与访问
AO3官网入口与中文阅读设置 AO3网页版使用与访问

本专题围绕 Archive of Our Own(AO3)官网入口展开,系统整理 AO3 最新可用官网地址、网页版访问方式、正确打开链接的方法,并详细讲解 AO3 中文界面设置、阅读语言切换及基础使用流程,帮助用户稳定访问 AO3 官网,高效完成中文阅读与作品浏览。

62

2026.02.02

热门下载

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

精品课程

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

共48课时 | 8.3万人学习

Excel 教程
Excel 教程

共162课时 | 15.1万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 2万人学习

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

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