热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C++动态库的制作和调用

原文链接:https:blog.csdn.netw_x_myselfarticledetails822526461、dll的有点代码复用是提高软件开发效率的重要途径。一般而言,只要

原文链接:https://blog.csdn.net/w_x_myself/article/details/82252646

1、dll的有点
代码复用是提高软件开发效率的重要途径。一般而言,只要某部分代码具有通用性,就可将它构造成相对独立的功能模块并在之后的项目中重复使用。比较常见的例子是各种应用程序框架,ATL、MFC等,它们都以源代码的形式发布。由于这种复用是“源码级别”的,源代码完全暴露给了程序员,因而称之为“白盒复用”。“白盒复用”的缺点比较多,总结起来有4点。 
暴露了源代码;多份拷贝,造成存储浪费; 
容易与程序员的“普通”代码发生命名冲突; 
更新功能模块比较困难,不利于问题的模块化实现; 
实际上,以上4点概括起来就是“暴露的源代码”造成“代码严重耦合”。为了弥补这些不足,就提出了“二进制级别”的代码复用。使用二进制级别的代码复用一定程度上隐藏了源代码,对于缓解代码耦合现象起到了一定的作用。这样的复用被称为“黑盒复用”。 

说明:实现“黑盒复用”的途径不只dll一种,静态链接库甚至更高级的COM组件都是。

2、ddl的创建
2.1、创建及注意事项
文件------>新建------>项目------>Win32控制台应用程序/Win32项目------>单击下一步------>应用程序类型选择DLL(图1)------>单击完成。
技术分享图片

 

 

 创建出来原始项目结构:

技术分享图片

 

 

 在附加选项中,选择空项目,生成的项目结构

技术分享图片

 

 

 注意:解决方案配置问题,win32平台生成的dll文件,只能被win32平台运行的项目调用:x64平台生成的dll文件,只能被x64平台运行的项目调用。

2.2、动态库制作方法
extern "C" _declspec(dllexport)与project2.h中的#ifdef.......endif是将C++函数导出,才会生成lib文件

2.2.1、方法一
通过定义C的接口函数对类方法进行封装,及定义全局变量,源码如下(此方法定义的类,还可以进行多项目联合编程):

Project1.h


#include "stdafx.h"
#include

#include
<string>
using namespace std;
class project1
{
public:
project1();
~project1();
void project1_name();
void project1_budget(int money);
bool project1_run();
int project1_numPeople();
string project_name;
};

Project1.cpp


#include "stdafx.h"
#include
"Project1.h"
project1::project1(){}
project1::
~project1(){}
project1 theApp;
//定义一个全局变量,方便被封装函数调用类的方法
void project1::project1_name()
{
cout
<<"项目名称为:"<<endl;
cout
< endl;
}
void project1::project1_budget(int money)
{
cout
< endl;
}
bool project1::project1_run()
{
return true;
}
int project1::project1_numPeople()
{
return 10;
}
extern "C" _declspec(dllexport) void name()
{
theApp.project1_name();
}
extern "C" _declspec(dllexport) void budget(int money)
{
theApp.project1_budget(money);
}
extern "C" _declspec(dllexport) bool run()
{
return theApp.project1_run();
}
extern "C" _declspec(dllexport) int numPeople()//对numPeople进行封装,需要使用关键字extern "C" _declspec(dllexport),运用关键字后,才会生产lib文件
{
return theApp.project1_numPeople();
}


2.2.2、方法二

将类的成员函数直接封装成C接口,源码如下

project2.h


#ifdef TESTDLL_EXPORTS
#define TESTDLL_API __declspec(dllexport)
#else
#define TESTDLL_API __declspec(dllimport)
#endif
#include

#include
<string>

using namespace std;
class project2
{
public:

project2();
~project2();
TESTDLL_API
void project2_name();
TESTDLL_API
void project2_budget(int money);//库制作不会报异常,但是传入参数会无效
//应修改成 static TESTDLL_API void project2_budget(int money);
TESTDLL_API bool project2_run();
TESTDLL_API
int project2_numPeople();
string project_name;
};

project2.cpp


project2::project2(){}
project2::
~project2(){}
void project2::project2_name()
{
project_name
=“项目2”;//1
cout <<"项目名称为:" << endl;
cout
<//2
//调用时,应该把1和2注销,原因未理解
}
void project2::project2_budget(int money)
{
cout
< endl;
}
bool project2::project2_run()
{
return true;
}
int project2::project2_numPeople()
{
return 20;
}

2.3、查看动态库生成的接口
运用的工具:单击Windows图标------>所有程序------>找到相应的Visual Studio文件夹------->选择Visual Studio tool(会打开文件夹)-------->寻找本机工具命令提示。切换到dll文件目录下,运行命令:dumpbin /EXPORTS 库名(例:dumpbin /EXPORTS Project2.dll)

方法一生成的动态库结构图:
技术分享图片

 

 

 方法二生成的动态库结构图:

技术分享图片


3、动态库的链接

3.1、显示链接

获取dll库的路径,无需配置环境,代码如下:


#include
#include

using namespace std;
void Display_Call_Project1_DLL()
{
typedef
void(*name)();
typedef
void(*budget)(int money);
typedef
bool(*run)();
typedef
int(*numPeople)();
HMODULE hDLL
= LoadLibrary("..\\..\\Make_Dll\\x64\\Debug\\Project1.dll");//dll的文件路径
if (hDLL == NULL)
{
cout
<<"动态库未找到" << endl;
return;
}
name n
= name(GetProcAddress(hDLL, "name"));//运用函数名
n();
budget b
= budget(GetProcAddress(hDLL, MAKEINTRESOURCE(1)));//运用序号调用,调用的为budget函数
b(2000);

run r
= run(GetProcAddress(hDLL, "run"));
cout
< endl;

numPeople np = numPeople(GetProcAddress(hDLL, "numPeople"));
cout
< endl;
FreeLibrary(hDLL);
}
//调用方法二生成的动态库
void Display_Call_Project2_DLL()
{
typedef
void(*name)();
typedef
void(*budget)(int money);
typedef
bool(*run)();
typedef
int(*numPeople)();
HMODULE hDLL
= LoadLibrary("..\\..\\Make_Dll2\\x64\\Debug\\Project2.dll");//dll的文件路径
if (hDLL == NULL)
{
cout
<<"动态库未找到" << endl;
return;
}
name n
= name(GetProcAddress(hDLL, "?project2_name@project2@@QEAAXXZ"));//运用函数名
n();
budget b
= budget(GetProcAddress(hDLL, MAKEINTRESOURCE(1)));//运用序号调用,调用的为budget函数
b(1000);

run r
= run(GetProcAddress(hDLL, MAKEINTRESOURCE(4)));
cout
< endl;

numPeople np = numPeople(GetProcAddress(hDLL, MAKEINTRESOURCE(3)));
cout
< endl;
FreeLibrary(hDLL);
}
int main()
{
Display_Call_Project2_DLL();
system(
"pause");
return 0;
}

注意:运用方法二,生成的动态库,成员函数必须设置静态成员函数,并且不能调用类的成员。否则入传参无效,并且调用类成员会报错。

3.2、隐式链接
必须配置环境:

项目->属性->配置属性->VC++ 目录-> 在“包含目录”里添加头文件project2.h所在的目录

项目->属性->配置属性->VC++ 目录-> 在“库目录”里添加头文件project2.lib所在的目录

项目->属性->配置属性->链接器->输入-> 在“附加依赖项”里添加“project2.lib”(若有多个 lib 则以空格隔开)

方法一生成的动态库,无法进行隐式链接。

隐式链接动态库的制作方法,必须在类函数中加上宏定义,源码如下:

project2.h


#ifdef TESTDLL_EXPORTS

#define TESTDLL_API __declspec(dllexport)
#else
#define TESTDLL_API __declspec(dllimport)
#endif
#include

#include
<string>

using namespace std;
class project2
{
public:

TESTDLL_API project2();
TESTDLL_API
~project2();
TESTDLL_API
void project2_name();
TESTDLL_API
void project2_budget(int money);
TESTDLL_API
bool project2_run();
TESTDLL_API
int project2_numPeople();
string project_name;
};

project2.cpp


#define TESTDLL_EXPORTS//不进行宏定义,或提示链接不一致,导致隐式调用失败
#include
"Project2.h"
project2::project2(){}
project2::
~project2(){}
void project2::project2_name()
{
//project_name = "项目2";
cout <<"项目名称为:";
//cout <
}
void project2::project2_budget(int money)
{
cout
< endl;
}
bool project2::project2_run()
{
return true;
}
int project2::project2_numPeople()
{
return 20;
}

 调用函数:

main.cpp


#include
#include

#include
"Project1.h"
#include
"project2.h"
using namespace std;
void Call_Project2_DLL()
{
project2 p;
p.project2_run();
p.project2_budget(
1000);

}
int main()
{

Call_Project2_DLL();
system(
"pause");
return 0;
}

注:提示找不到dll库时,将dll库放在main.cpp同级目录下


推荐阅读
  • 如果应用程序经常播放密集、急促而又短暂的音效(如游戏音效)那么使用MediaPlayer显得有些不太适合了。因为MediaPlayer存在如下缺点:1)延时时间较长,且资源占用率高 ... [详细]
  • 本文详细介绍了如何解决DNS服务器配置转发无法解析的问题,包括编辑主配置文件和重启域名服务的具体步骤。 ... [详细]
  • [转]doc,ppt,xls文件格式转PDF格式http:blog.csdn.netlee353086articledetails7920355确实好用。需要注意的是#import ... [详细]
  • 网络爬虫的规范与限制
    本文探讨了网络爬虫引发的问题及其解决方案,重点介绍了Robots协议的作用和使用方法,旨在为网络爬虫的合理使用提供指导。 ... [详细]
  • [c++基础]STL
    cppfig15_10.cppincludeincludeusingnamespacestd;templatevoidprintVector(constvector&integer ... [详细]
  • 网站访问全流程解析
    本文详细介绍了从用户在浏览器中输入一个域名(如www.yy.com)到页面完全展示的整个过程,包括DNS解析、TCP连接、请求响应等多个步骤。 ... [详细]
  • 微软推出Windows Terminal Preview v0.10
    微软近期发布了Windows Terminal Preview v0.10,用户可以在微软商店或GitHub上获取这一更新。该版本在2月份发布的v0.9基础上,新增了鼠标输入和复制Pane等功能。 ... [详细]
  • Framework7:构建跨平台移动应用的高效框架
    Framework7 是一个开源免费的框架,适用于开发混合移动应用(原生与HTML混合)或iOS&Android风格的Web应用。此外,它还可以作为原型开发工具,帮助开发者快速创建应用原型。 ... [详细]
  • 本文介绍了如何使用 CMD 批处理脚本进行文件操作,包括将指定目录下的 PHP 文件重命名为 HTML 文件,并将这些文件复制到另一个目录。 ... [详细]
  • 两个条件,组合控制#if($query_string~*modviewthread&t(&extra(.*)))?$)#{#set$itid$1;#rewrite^ ... [详细]
  • 开机自启动的几种方式
    0x01快速自启动目录快速启动目录自启动方式源于Windows中的一个目录,这个目录一般叫启动或者Startup。位于该目录下的PE文件会在开机后进行自启动 ... [详细]
  • 解决Parallels Desktop错误15265的方法
    本文详细介绍了在使用Parallels Desktop时遇到错误15265的多种解决方案,包括检查网络连接、关闭代理服务器和修改主机文件等步骤。 ... [详细]
  • 解决 Windows Server 2016 网络连接问题
    本文详细介绍了如何解决 Windows Server 2016 在使用无线网络 (WLAN) 和有线网络 (以太网) 时遇到的连接问题。包括添加必要的功能和安装正确的驱动程序。 ... [详细]
  • 使用Jsoup解析并遍历HTML文档时,该库能够高效地生成一个清晰、规范的解析树,即使源HTML文档存在格式问题。Jsoup具备强大的容错能力,能够处理多种异常情况,如未闭合的标签等,确保解析结果的准确性和完整性。 ... [详细]
  • WinMain 函数详解及示例
    本文详细介绍了 WinMain 函数的参数及其用途,并提供了一个具体的示例代码来解析 WinMain 函数的实现。 ... [详细]
author-avatar
00我就是我00乐乐
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有