日期:2014-05-16  浏览次数:21117 次

C#调用C++的DLL乱码, 函数原型是:char *fuc(char *a,char *b)
在C语言的标准库函数中,有strcat()函数,它的作用是连接两个字符串,原型如下:

char *strcat(char *strDestination, const char *strSource)

作用是:在 Dest 的后面接上 Src 指向的字符串,把两个字符串连接起来。

我自己实现了 my_strcat(char *dst, char *src)函数,代码如下:
extern "C"
{
__declspec(dllexport) char *my_strcat(char *dst, char *src)
{
char *old_pos = dst;
while (*dst)
{
dst++;
}
while (*dst++ = *src++)
{
;
}
return old_pos;
}
}

上面的函数经过控制台测试,是正常的,然后使用VS2010编译得到 *.dll 动态连接库文件。
在C#中的代码如下:

//导入刚刚生成的 DLL ,函数入口是 my_strcat ,调用方式是 cdecl ,字符集是 unicode

[DllImport("*.dll", EntryPoint = "my_strcat", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Unicode)]
public static extern string my_strcat(string a, string b);     

//在点击事件中把文本框1中的字符串,文本框2中的字符串连接起来,然后把连接的字符串放在文本框3中。
textBox3.Text = my_strcat(textBox1.Text, textBox2.Text);

但是点击按钮后文本框3中的内容是乱码,请教,如何解决C#调用DLL里,针对 C/C++ 中的char * 函数(char *,char *)形参的调用?并且能正常返回一个字符串?


------解决方案--------------------

// 危险代码!!
int i = 0;
char src[] = "hello";
my_strcat(src, "buffer overflow");
printf("i=%x", i);   //i=776f6c66


比如以上代码,src缓冲区的实际大小只有6(五个字符加上一个零休止)。
把另外一个字符串接在它后面,就可能修改了不是src缓冲区的其他数据。
比如以上试验中,i就被覆盖了(不同编译器可能有不同内存布局,可能有不同结果)。如果覆盖的不只是int i,而是影响了线程堆栈,或者其他重要信息,程序就可能崩溃了,甚至可能把你C盘格式化。

你在C#中用string来传入dst,这种情况下dst一点都没有预留空间,几乎肯定要缓冲溢出。
因此,解决问题的关键就要要保证dst主够大,比如:
StringBuilder dst = new StringBuilder(1024);  //预留1024个字符
dst.Append("hello");
my_strcat(des, " world");

但是,问题是要预留多大的空间?1024会不会浪费?1024会不会太小?
更好的解决方法,就是明确指示缓冲区的大小。比如,换成如下导出:

void my_str(char* dst, const char* src, int dst_length)
{
  int offset = strnlen(dst, dst_length);
  for(int i = offset; i < dst_length && *src; i++)
  {
    dst[i] = *src++;
  }
}

其中src用const char*修饰,const表示不会被改变,因此可以安全的用C# string传入。
其中dst则可以被改变,它的大小用dst_length来限制。

这也是为什么strcat被标记被不安全代码,而strncat被要求使用的原因。
------解决方案--------------------
vc项目右键设置多字节的。