微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

如何使用__declspec(dllexport / import)从DLL /导出/导入C结构到控制台应用程序

这是我第一次处理DLL.在MSDN文档之后,我创建了一个文件fooExports.h,其中宏根据预处理器定义定义:
#ifdef FOODLL_EXPORTS
    #define FOO_API __declspec( dllexport )
#else
    #define FOO_API __declspec( dllimport )

我的目的是在我的DLL实现以及控制台应用程序中使用此标头.到目前为止,导入和导出功能工作正常.当我尝试导出一个已经定义的结构作为其中一个导出函数的参数时,问题就出现了.例如,在前面提到的头文件中,我声明了FOO_API void foo(FooParams * args),而args是一个定义如下的结构:

typedef struct FooParams
{
    char *a;
    char *b;
    void *whatever; //some other type
} FooParams;

此结构必须在foo.h中定义,而不是在fooExports.h中定义.是否有任何方法可以导出此结构而不将其从原始头文件删除(考虑到我希望将导出/导入集中在fooExports.h中).
这样做会有什么更好的方法? DLL都是C以及使用它的客户端应用程序.

如果客户端对FooParams的唯一用途是获取从DLL函数返回的指针并将这些指针传递给其他DLL函数,则可以使其成为“不透明类型”:Put
typedef struct FooParams FooParams;

在fooExports.h中. FOO_API宏不属于该声明. opaque类型意味着客户端代码不能:

>创建FooParams类型的任何变量(但是FooParams * ptr = NULL;没关系).
>对FooParams的任何成员做任何事情.
>查找sizeof(FooParams) – 因此无法正确地为一个或多个FooParams对象提供malloc空间.

您也不能#define对客户端可见的宏执行上述任何操作.所以你的DLL需要有一个或多个“构造函数”或“工厂”函数,可能就像

FOO_API FooParams* CreateFooParams(const char * input);

定义匹配的“析构函数函数也是一种很好的做法

FOO_API void DestroyFooParams(FooParams * p);

即使定义像{free(p);因为如果在DLL内部分配的内存被外部的代码释放,反之亦然(因为并非所有的Windows代码都使用相同的malloc和free定义),因此有时会出现问题.

如果所有这些都太极端,唯一的另一个选择是在导出的头文件中放入或#include结构定义,并使其对客户端可见.没有它,除了传递指针之外,任何对FooParams做某事的尝试都是不可能的,因为编译器不会知道FooParams中的内容.编译器(与链接器相对)仅从命令行参数和#include-d文件获取信息,而不是从库或DLL获取信息.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Windows注册表操作基础代码 Windows下对注册表进行操作使用的一段基础代码Reg.h:#pragmaonce#include<assert.h>#include<windows.h>classReg{HKEYhkey;public:voidopen(HKEYroot
黑客常用WinAPI函数整理之前的博客写了很多关于Windows编程的内容,在Windows环境下的黑客必须熟练掌握底层API编程。为了使读者对黑客常用的Windows API有个更全面的了解以及方便日后使用API方法的查询,特将这些常用的API按照7大分类进行整理如下,希望对大家的学习有所帮助。一
一个简单的Windows Socket可复用框架说起网络编程,无非是建立连接,发送数据,接收数据,关闭连接。曾经学习网络编程的时候用Java写了一些小的聊天程序,Java对网络接口函数的封装还是很简单实用的,但是在Windows下网络编程使用的Socket就显得稍微有点繁琐。这里介绍一个自己封装的一
Windows文件操作基础代码 Windows下对文件进行操作使用的一段基础代码File.h,首先是File类定义:#pragmaonce#include<Windows.h>#include<assert.h>classFile{HANDLEhFile;//文件句柄publ
Winpcap基础代码 使用Winpcap进行网络数据的截获和发送都需要的一段代码:#include<PCAP.H>#pragmacomment(lib,"wpcap.lib")//#pragmacomment(lib,"ws2_32.lib")#
使用vbs脚本进行批量编码转换 最近需要使用SourceInsight查看分析在Linux系统下开发的项目代码,我们知道Linux系统中文本文件默认编码格式是UTF-8,而Windows中文系统中的默认编码格式是Gb2312。系统内的编码格式有所区别倒无伤大雅,关键的是SourceInsigh...
缓冲区溢出攻击缓冲区溢出(Buffer Overflow)是计算机安全领域内既经典而又古老的话题。随着计算机系统安全性的加强,传统的缓冲区溢出攻击方式可能变得不再奏效,相应的介绍缓冲区溢出原理的资料也变得“大众化”起来。其中看雪的《0day安全:软件漏洞分析技术》一书将缓冲区溢出攻击的原理阐述得简洁
Windows字符集的统一与转换一、字符集的历史渊源在Windows编程时经常会遇到编码转换的问题,一直以来让刚接触的人摸不着头脑。其实只要弄清Win32程序使用的字符编码方式就清楚了,图1展示了一个Win32控制台项目的属性中的字符集选项。这里有两个不同的字符集:一个是Unicode字符集,另一个
远程线程注入引出的问题一、远程线程注入基本原理远程线程注入——相信对Windows底层编程和系统安全熟悉的人并不陌生,其主要核心在于一个Windows API函数CreateRemoteThread,通过它可以在另外一个进程中注入一个线程并执行。在提供便利的同时,正是因为如此,使得系统内部出现了安全
windows系统启动项怎么打开