- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
环境是Windows,MS Visual Studio2012。如何在C(不是C++)应用程序中使用以C++编写的DLL库?在我的DLL中,我使用了 namespace ,类等。例如,我写了two simple projects:
最佳答案
解决方案是在C++中将DLL编写为C接口(interface)(作为DLL的一部分,或作为单独的DLL),然后从C代码中使用该C接口(interface)。
确保您的C接口(interface)包含都是extern "C"
的函数。
编辑:正如SigTerm在注释中所指出的那样,将函数stdcall
(默认情况下不是默认的,但是有些人会从WinAPI函数中复制它)不是一个好主意,因为在尝试使用WinAPI函数时会造成麻烦。其他语言。
以下是挑战以及如何解决这些挑战:
您的类(class)在 namespace 中
通常,您只需声明不透明的struct ClassName*
即可在C中导出C++类。但是,这对于命名空间中的类是不可能的,因为C不知道命名空间的概念。有几种解决方法:
static_cast
更改为实际类型。intptr_t
中的stdint.h
,因为无论您在什么平台上,它都保证足够大。否则,为了安全起见,最好使用long
。void*
略强,它是任何曾经处理过OS文件处理等底层功能的C程序员所熟悉的界面风格。void*
或整数句柄作为成员的C结构。void*
或raw句柄那么容易),类型安全(只要C代码不会与struct成员混淆)。std::string
std::string
一起使用,并且您当然不想做将接口(interface)包装到C中的工作(您的C接口(interface)用户无论如何都不希望您这样做)。因此,必须在C接口(interface)中使用
char*
/
char const*
。
char const*
,然后让
std::string
的构造函数完成其余工作即可。
c_str
的结果,因为当临时
std::string
被破坏时,它将留下一个悬空的指针。因此,您基本上有以下选择:
std::string
(这样它就不会在返回时被释放),将函数调用的结果复制到其中,然后返回c_str
。strncpy
将字符串数据复制到字符数组。enum
成员
enum
definitios为标识符提供相同的值,因此只需将类内枚举中的值强制转换为全局枚举,然后再将其转换回即可。唯一的问题是,每当您更新类内枚举时,都必须记住也要更新类外枚举。
std::vector
,另一种选择是给出指向 vector 第一个元素的指针。但是,这些将在下次调整 vector 大小时失效,因此仅在非常有限的情况下才是一种选择。
print
函数,该函数需要
std::ostream
和对对象的引用。
std::ostream
对C代码不可用。在C语言中,使用
FILE*
访问文件。我认为最好的选择是使用iostream类,该类围绕
FILE*
并允许您从一个iot_code构造。我不知道MSVC++是否提供了该功能,但如果不提供,Boost具有AFAIK这样的流。
FILE*
和一个对象句柄(或void指针,或您选择的用于在C中表示对象的任何选项),从FILE指针构造一个临时的ostream对象,从该句柄中提取指向该对象的指针,然后在指向的对象上调用
print
。
/* somelib_c_interface.h */
#ifndef SOMELIB_C_INTERFACE_H
#define SOMELIB_C_INTERFACE_H
#ifdef __cplusplus
extern "C" {
#endif
#include <stdint.h>
#include <stdlib.h>
/* typedef for C convenience */
typedef struct
{
intptr_t handle;
} some_namespace_info;
typedef struct
{
intptr_t handle;
} some_namespace_employee;
/* error codes for C interface */
typedef int errcode;
#define E_SOMELIB_OK 0 /* no error occurred */
#define E_SOMELIB_TRUNCATE -1 /* a string was created */
#define E_SOMELIB_OUT_OF_MEMORY -2 /* some operation was aborted due to lack of memory */
#define E_SOMELIB_UNKNOWN -100 /* an unknown error occurred */
/* construct an info object (new)
errcode some_namespace_info_new(some_namespace_info* info, char const* name, char const* number);
/* get rid of an info object (delete) */
errcode some_namespace_info_delete(some_namespace_info info);
/* Some_namespace::Info member functions */
errcode some_namespace_info_get_name(some_namespace_info info, char* buffer, size_t length);
errcode some_namespace_info_set_name(some_namespace_info info, char const* name);
errcode some_namespace_info_get_number(some_namespace_info info, char* buffer, size_t length);
errcode some_namespace_info_set_number(some_namespace_info info, char const* name);
/* the employee class */
/* replicated enum from employee */
enum some_namespace_employee_sex { male, female };
errcode some_namespace_employee_new(some_namespace_employee* employee,
char const* name, char const* surname, int age,
some_namespace_employee_sex sex);
errcode some_namespace_employee_delete(some_namespace_employee employee);
errcode some_namespace_employee_get_name(some_namespace_employee employee, char* buffer, size_t length);
errcode some_namespace_employee_set_name(some_namespace_employee employee, char const* name);
errcode some_namespace_employee_get_surname(some_namespace_employee employee, char* buffer, size_t length);
errcode some_namespace_employee_set_surname(some_namespace_employee employee, char const* name);
/* since ages cannot be negative, we can use an int return here
and define negative results as error codes, positive results as valid ages
(for consistency reason, it would probably be a better idea to not do this here,
but I just want to show another option which sometimes is useful */
int some_namespace_employee_get_age(some_namespace_employee employee);
errcode some_namespace_employee_set_age(some_namespace_employee employee, int age);
errcode some_namespace_employee_get_set(some_namespace_employee employee,
enum some_namespace_employee_sex* sex);
errcode some_namespace_employee_set_sex(some_namespace_employee employee,
enum some_namespace_employee_sex sex);
typedef struct
{
intptr_t handle;
} info_iter;
info_iter_delete(info_iter iterator);
some_namespace_info info_iter_get(info_iter iterator);
void info_iter_next(info_iter iterator);
bool info_iter_finished(info_iter iterator);
info_iter some_namespace_employee_phones(some_namespace_employee employee);
info_iter some_namespace_employee_emails(some_namespace_employee employee);
info_iter some_namespace_employee_sites(some_namespace_employee employee);
errcode some_namespace_print(FILE* dest, some_namespace_employee employee);
#ifdef __cplusplus
}
#endif
#endif // defined(SOMELIB_C_INTERFACE_H)
关于c++ - 如何在C应用程序中使用C++ DLL?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17393134/
我有一个 Windows Service(一个发布版本),我替换了一个在 Debug模式下构建的 DLL 并尝试启动服务。它引发了错误无法加载文件或程序集“名称”或其依赖项之一。尝试加载格式不正确的程
我使用 Microsoft Visual Basic 6.0 Enterprise Edition SP6 开发了一个 VB6 应用程序。 我已将可执行文件复制到未安装 VB6 的计算机 C。 我已在
我有很多小的 DLL,我想将它们组合成一个大的(更)DLL(如 suggested here)。我可以通过合并我的项目来做到这一点,但我想要一种不那么干扰的方式。 可以将多个 DLL 合并为一个单元吗
我想将非托管 DLL 和图像文件与托管 DLL 合并。我该怎么做?可能吗? 最佳答案 很有可能。这可以通过 bxilmerge 来完成此解决方案可以将非托管 DLL 和图像或其他文件与托管 DLL 合
如何在Windows Server 2003上安装msvcr71.dll,这是我的软件所需的。我真的不想将此dll复制到system32文件夹,因为它可能会破坏此目标系统。 最佳答案 只需将其复制到程
我最近遇到了一个安装在我的系统上的 DLL,Dependancy Walker(以及我尝试过的所有其他实用程序)说按名称或顺序导出为零,但文件大小约为 4mb。我认为 DLL 的唯一目的是导出供其他代
我终于让我的本地主机在本地显示该站点。一切似乎都在朝着这个方向努力。我的下一步是当网站使用 ActiveX.dll 中的函数时,实际上能够从 VB6 IDE 进入代码 更新: 我更新了代码并删除了我在
首先,我指的是Windows环境和VC++编译器。 我想要做的是重建一个Vc++ dll并与已经链接到该lib的exe保持兼容性,而不必重建该exe或使用LoadLibrary动态加载该dll。换句话
我以前这样做过,我不记得我是从网上下载的 DLL 还是其他东西,但我不想感染病毒。我需要访问这个命名空间,以便我可以拥有 Webbrowswer 控件不提供的额外功能。 如何准确添加 Com 引用。还
我正在尝试为依赖于 ghostscript 的库创建 Nuget 包,因此引用 gsdll32.dll - 一个非托管库。我不能只包含一个标准的 dll 引用。我应该把它放在 nuget 目录结构中的
如果我有 Windows 可执行文件,我如何找出它将加载哪些 dll? 我只是谈论哪些将静态加载,而不是那些可能使用 LoadLibrary 等动态加载的内容。 最佳答案 dumpbin是VC++自带
这可能是一个非常菜鸟的问题,但在当今的网络应用程序开发世界中,许多程序员不需要过多处理 dll,因此也懒得去了解它们的用途。 那么什么是dll? 它有什么用? 它是如何工作的? 如何创建一个? 在什么
所以我刚刚开始使用 OpenTK,并将此代码放在一个继承 GameWindow 类的类中: protected override void OnRenderFrame(FrameEventArgs e
如何调试未由 java 应用程序加载的 dll。 场景是这样的:我的 java 应用正在加载正在使用另一个 dll 的 jni.dll,而那个 dll 正在使用另一个 dll。 javajni.dll
我将“更好”放在引号中,因为这是一个定性问题。几年来我一直在编写 COM DLL,直到最近才发现并成功使用了带有 Typelib 概念的标准 DLL。 使用 COM DLL 代替 DLL+Typeli
这种用户机器可能没有msvcp100.dll、msvcp100.dll之类dll的情况怎么处理?我不希望我的软件因为这种错误而无法安装在用户的机器上。我一直在考虑要么找到一个工具并将每个需要的 dll
在 VS2012 中使用 C# .NET 和 COM 互操作,我正在开发一个用于其他几个程序的公共(public)库。为了简化集成,我想将整个库缩减为一个 DLL。该库的特点之一是本地化。它有包含多种
我在 .Net 中使用 FieldTalk Modbus。当我运行应用程序时,我在为 MbusTcpMasterProtocol 创建对象时遇到错误。 MbusTcpMasterProtocol mb
我想指出,我知道如何向/从 GAC 添加/删除程序集。我要问的是,是否有人可以从技术角度向我解释它是如何工作的。将 dll 放在那里有什么意义 - 从节省资源的角度来看。 亲切的问候 最佳答案 将东西
我继承了一个传统的经典 ASP 应用程序,它使用 VB6 ActiveX DLL 来执行业务逻辑。 我想跟踪加载和卸载 DLL 的点。有没有办法在 VB6 DLL 中拦截这些事件? 在相关说明中,Cl
我是一名优秀的程序员,十分优秀!