c#无法执行带有内存集的c++dll
本文关键字:内存 c++dll 行带 执行 | 更新日期: 2023-09-27 17:58:34
我正在为c#构建一个c++库。然而,当里面有内存集或内存时,我的程序就死了。下面是代码:
c#
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Runtime.InteropServices;
namespace CsharpCallDll
{
public class dllfunction
{
[DllImport("dllgen.dll", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Ansi)]
public static extern int calData(ref double data, int data_size,
ref double info, int info_size, ref char result,int max_result_size,ref int realResultSize);
}
class Program
{
static void Main(string[] args)
{
int data_size = 2;
double[] data = new double[data_size];
int info_size = 3;
double[] info = new double[info_size];
int max_result_size = 1000;
char[] result = new char[max_result_size];
int real_result_size = 0;
data[0] = 1;
data[1] = 2;
info[0] = 1;
info[1] = 2;
info[2] = 3;
unsafe
{
dllfunction.calData(ref data[0], data_size,
ref info[0], info_size, ref result[0], max_result_size,ref real_result_size);
Console.Write(result + "'n");
}
Pause();
}
public static void Pause()
{
Console.Write("Press any key to continue . . . ");
Console.ReadKey(true);
}
}
}
c++
MG_API int calData(double *data, int dataSize,
double *info, int infoSize, char* result, int maxresultSize, int* realResultSize)
{
int errorCode = 0;
if (dataSize == 0 || infoSize == 0)
{
errorCode = 1;
}
string resultArray = "";
double* tempvalue = new double;
string isnormal;
string abnromaltype;
int resultSiz = 0;
for (int i = 0; i < 15; i++)
{
srand((int)time(NULL)+i);
*tempvalue = (double)Random(10);
if (*tempvalue > 8) {
isnormal = "false";
abnromaltype = abnromalTypeS[0];
}else if (*tempvalue < 2) {
isnormal = "false";
abnromaltype = abnromalTypeS[1];
}else{
isnormal = "true";
abnromaltype = abnromalTypeS[2];
}
resultArray = appendResultIntoString(resultArray, dataNameS[i], *tempvalue,
dataUnitS[i], isnormal, abnromaltype);
}
*realResultSize = resultArray.size();
// print the result
cout << "this is the resultsize ";
cout << *realResultSize << endl;
// result = (char*) malloc(*resultSize);
memset(result, 0, maxresultSize);
// memcpy(result, resultArray.c_str(), *realResultSize);
delete tempvalue;
return errorCode;
}
resultArray是一个字符串结果。
我的推测是:1.当c++代码中存在memcpy、memset时,c#程序将在那里消亡。发生了什么?
顺便说一下:2.有没有一种方法可以将一个动态长度的字符串从c++发送到c#3.有没有一种方法可以将字符串[]从c++发送到c#
谢谢你们!
..., ref char result,int max_result_size,ref int realResultSize
char*
参数在C和C++中是不明确的。可能是指对单个字符的引用(C#中的ref-char)或对数组的引用(C#中的char[])。这在这些语言中并不重要,尽管你肯定会像在这里出错一样出错,但它在C#中产生了很大的差异。在C中,通过传递一个char&而不是char[]&并且memset()调用将损坏调用方的堆栈。同样的事情也发生在这里。
由于pinvokemarshaller不知道它实际上应该传递一个数组引用,并且char
类型与C代码不兼容,因为它在C#中是2个字节,在C中是1个字节,所以marshaller复制单个字符以将其转换为字节。您的memset()调用现在会破坏pinvokemarshaller为该单个字节分配的内存。结果是不可预测的,你只有在幸运的情况下才能获得AVE。
它不知道的另一件事是,它必须将数组复制回来。除了不知道它是一个数组之外,它也不知道它的长度。你必须要求它,默认情况下,pinvokemarshaller不会将数组复制回来以避免这样做的成本
告诉pinvokemarshaller这是一个需要复制回的数组,如下所示:
..., [Out][MarshalAs(UnmanagedType.LPArray, SizeParamIndex(6)] char[] result
[Out]属性要求将数组元素复制回,SizeParamIndex属性表示当整理器需要知道需要复制多少元素时,应该在哪里查找。
通过不强制pinvokemarshaller转换数组元素,它既简单又高效:
..., byte[] result
不再需要在[Out]和[MarshalAs]属性方面提供帮助。由于不再需要转换,pinvokemarshaller可以简单地固定数组并将指针传递给它的第一个元素。您的C代码现在直接写入GC堆存储。有了max_result_size参数是件好事,避免GC堆损坏非常重要。请确保传递数组的Length属性。
请注意,同样的情况也适用于data
和info
参数。没有那么致命,因为它们不需要转换。将它们声明为double[]
,而不是引用
除了Hans-Passant的建议之外,另一种可能性是用fixed
语句显式地固定变量。
unsafe
{
fixed (double* d = data)
{
fixed (double* i = info)
{
fixed (int* rs = &real_result_size)
{
dllfunction.calData((IntPtr)d, data_size, (IntPtr)i, info_size, result, max_result_size, (IntPtr)rs);
}
}
}
}