c#程序调用c++开发dll库

最近算法组同事开发一个接口,如获取名称:

cpp 复制代码
extern "C" __declspec(dllexport) void GetName(std::string& name);

打包成 dll 库后,供我这边 c# 项目中调用如下:

cs 复制代码
[DllImport("Test.dll", EntryPoint = "GetName", ExactSpelling = false, CallingConvention = CallingConvention.Cdecl)]
public static extern void GetName(out StringBuilder name);

// 获取名称
public static string GetName()
{
    StringBuilder name = new StringBuilder(256);
    GetName(out name);

    return name.ToString();
}

正常情况下,并没有什么问题。但是如果连续两次调用 dll 库中方法,程序必报错,提示尝试读取或写入受保护的内存

开始觉得是因为传参被修改导致,沟通后将名称通过函数返回值返回,避免参数修改。

cpp 复制代码
extern "C" __declspec(dllexport) std::string GetName();

对应的,项目中程序也调整为:

cs 复制代码
[DllImport("Test.dll", EntryPoint = "GetName", ExactSpelling = false, CallingConvention = CallingConvention.Cdecl)]
public static extern string GetName();

// 获取名称
public static string GetName()
{
    string name = GetName();

    return name;
}

运行后程序直接报错了,一番查询后大概率是因为 c# 中的 string 类型和 c++ 中的 std::string 类别不匹配导致。同时让同事改为返回 const char* 试试。即:

cpp 复制代码
extern "C" __declspec(dllexport) const char* GetName();

调用如下:

cs 复制代码
[DllImport("Test.dll", EntryPoint = "GetName", ExactSpelling = false, CallingConvention = CallingConvention.Cdecl)]
public static extern IntPtr GetName();

public static string GetName()
{
    IntPtr ptr = GetName();
    string name = Marshal.PtrToStringAnsi(ptr);

    return name;
}

至此,问题才得以解决。在正确返回信息的同时,即使连续调用也依然稳定可行。

相关推荐
感哥4 小时前
C++ 多态
c++
沐怡旸11 小时前
【底层机制】std::string 解决的痛点?是什么?怎么实现的?怎么正确用?
c++·面试
mudtools12 小时前
.NET驾驭Word之力:理解Word对象模型核心 (Application, Document, Range)
c#·.net
River41614 小时前
Javer 学 c++(十三):引用篇
c++·后端
感哥17 小时前
C++ std::set
c++
侃侃_天下17 小时前
最终的信号类
开发语言·c++·算法
博笙困了18 小时前
AcWing学习——差分
c++·算法
大飞pkz18 小时前
【设计模式】C#反射实现抽象工厂模式
设计模式·c#·抽象工厂模式·c#反射·c#反射实现抽象工厂模式
青草地溪水旁18 小时前
设计模式(C++)详解—抽象工厂模式 (Abstract Factory)(2)
c++·设计模式·抽象工厂模式
青草地溪水旁18 小时前
设计模式(C++)详解—抽象工厂模式 (Abstract Factory)(1)
c++·设计模式·抽象工厂模式