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 *)形参的调用?并且能正常返回一个字符串?
19 个解决方案
#1
后来,我又在论坛上找到第二个版本如下:
结果不是乱码了,但是只是把两个字符串的首字母连接起来了,盼好心人能指点一二,给出代码以供参考,并能指出我错在哪里。谢谢了。
参考资料如下:
http://blog.csdn.net/xinyaping/article/details/7288164
在此特别感谢金辉软件技术公司的唐先生对我的帮助,谢谢所有帮助过我的人。
[DllImport("*.dll", EntryPoint = "my_strcat", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Unicode)]
public static extern IntPtr my_strcat(string a, string b);
IntPtr pStr = my_strcat(textBox1.Text, textBox2.Text);
textBox3.Text = Marshal.PtrToStringAnsi(pStr);
结果不是乱码了,但是只是把两个字符串的首字母连接起来了,盼好心人能指点一二,给出代码以供参考,并能指出我错在哪里。谢谢了。
参考资料如下:
http://blog.csdn.net/xinyaping/article/details/7288164
在此特别感谢金辉软件技术公司的唐先生对我的帮助,谢谢所有帮助过我的人。
#2
顺便再问一下:
在C#中,如果调用 FILE *fopen(“路径/文件名”, “模式”)这个 fopen 库函数?它返回的是一个文件流指针,在C#中如何操作该指针?若是有高手贴出代码,就太感谢了。
在C#中,如果调用 FILE *fopen(“路径/文件名”, “模式”)这个 fopen 库函数?它返回的是一个文件流指针,在C#中如何操作该指针?若是有高手贴出代码,就太感谢了。
#3
应该是CharSet = CharSet.Unicode这个出了问题,一般来说c/c++缺省的字符集是CharSet.ASCII,你用Unicode传递自然会有乱码,
#4
文件流是c库函数才能使用的东西,c#是没法用的,除非你把c库函数封装成dll,但是没人会这么做,
不要想在c#中操作库函数的流,
不要想在c#中操作库函数的流,
#5
你的C实现会缓存溢出,严重情况下会程序崩溃。
#6
这类问题最好不要去鼓励,相反地应该警告程序员“不要随便使用c++的dll”。
实际上,你应该让c++程序员(而不是自己)去提供.net封装的dll,并且经过严格测试其不会产生bug特别是泄露。
实际上,你应该让c++程序员(而不是自己)去提供.net封装的dll,并且经过严格测试其不会产生bug特别是泄露。
#7
我按你的办法使用了其它字符集,调试的时候VS2010提示: 尝试读取或写入受保护的内存。这通常指示其他内存已损坏。
请问还有其它的办法吗?谢谢了,我主要是学纯C开发,C#是我本来就会一些的,而且在未来的工作中,会很少用C#,大多数都是用C的多,我想使用DLL只是想把C的代码复用,要不然的话要用C#重新写一遍。
乱码的问题纠结了我好久,好想找个人帮我调试一下C#的工程文件,我的 QQ:86389981
盼高手能指点迷津。
#8
嗯,内存泄露的问题确实要注意,不过在写代码的时候注意free就可以使用了吧?我主要是想着把这个C的代码复用,在C#中也可以使用该项功能。如果可能,可不可以帮我看看我的代码哪里出了问题呢?谢谢你。
#9
如果可能,可不可以告诉我在哪个地方应该注意?我是新手。。。
我是第一次发贴,我觉得尽可能在贴子说出自己的想法和遇到的问题。
这样才会有高手看到,并指出自己的不对。
#10
楼主的C版本代码中的 my_strcat 函数有个 bug,你自己发现了吗?那就是缓冲区溢出问题。在你的代码中没有判断des参数的缓冲区是否够用就进行了连接。!
至于C#调用的代码,我还没学那么深,暂时不会。。、
至于C#调用的代码,我还没学那么深,暂时不会。。、
#11
// 危险代码!!
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被要求使用的原因。
#12
可用c++cli封装一层在给c#调用即可了
#13
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
//导入刚刚生成的 DLL ,函数入口是 my_strcat ,调用方式是 cdecl ,字符集是 unicode
[DllImport("test.dll", EntryPoint = "my_strcat", CharSet = CharSet.Ansi)]
public static extern string my_strcat(string a, string b);
private void button1_Click(object sender, EventArgs e)
{
//在点击事件中把文本框1中的字符串,文本框2中的字符串连接起来,然后把连接的字符串放在文本框3中。
textBox3.Text = my_strcat(textBox1.Text, textBox2.Text);
}
}
}
extern "C"
{
__declspec(dllexport) char *my_strcat(char *dst, char *src)
{
char *old_pos = dst;
while (*dst)
{
dst++;
}
while (*dst++ = *src++)
{
;
}
return old_pos;
}
}
vs2008调用成功。
#14
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
//导入刚刚生成的 DLL ,函数入口是 my_strcat ,调用方式是 cdecl ,字符集是 unicode
[DllImport("test.dll", EntryPoint = "my_strcat", CharSet = CharSet.Ansi)]
public static extern string my_strcat(string a, string b);
private void button1_Click(object sender, EventArgs e)
{
//在点击事件中把文本框1中的字符串,文本框2中的字符串连接起来,然后把连接的字符串放在文本框3中。
textBox3.Text = my_strcat(textBox1.Text, textBox2.Text);
}
}
}
extern "C"
{
__declspec(dllexport) char *my_strcat(char *dst, char *src)
{
char *old_pos = dst;
while (*dst)
{
dst++;
}
while (*dst++ = *src++)
{
;
}
return old_pos;
}
}
vs2008调用成功。
呵呵,感谢你的回复,我在论坛其它的贴子中看到你回复乱码的问题。
我分别用 VS2010 VS2008 VS2012 三种版本都测试过你的代码
均没有通过,是否哪个地方需要设置?比如工程属性?
我的 QQ 86389981 可以把你的工程文件传给我吗?
#15
dest 的空间大小需要调用方来保证,像strcpy / strncpy 也是一样的,strcpy 会强行把字符串末尾加个尾零符,这样会改变字符串的,所以都建议用 strncpy, 其实可以使用malloc动态分配内存,这都是后话了。
C的代码在C#里面复用,我纠结了很久,也没有搞明白,为什么是乱码?到底是哪里出了问题。我本来是想做一个计算器(在控制台中测试已实现),输入表达式字符串,然后根据括号和运算符优先级,计算出结果,想象中用C#做界面,用C在后台去处理字符串和算法。
C中没有string类,处理字符串都是指针或者是数组,如何把字符串在C#和C之间传递成了重中之重。如果把表达式的字符串全部用C#的函数处理后计算,就失去了我本意了。虽然不鼓励这样做,但是我只是做个试验,这样做能不能成功?所以就写了这个strcat函数来测试,能不能返回指针,并且形参也是指针。
如果能接收一个char指针,处理完后返回一个char指针,那么做一个表达式计算器就可能。
C的代码在C#里面复用,我纠结了很久,也没有搞明白,为什么是乱码?到底是哪里出了问题。我本来是想做一个计算器(在控制台中测试已实现),输入表达式字符串,然后根据括号和运算符优先级,计算出结果,想象中用C#做界面,用C在后台去处理字符串和算法。
C中没有string类,处理字符串都是指针或者是数组,如何把字符串在C#和C之间传递成了重中之重。如果把表达式的字符串全部用C#的函数处理后计算,就失去了我本意了。虽然不鼓励这样做,但是我只是做个试验,这样做能不能成功?所以就写了这个strcat函数来测试,能不能返回指针,并且形参也是指针。
如果能接收一个char指针,处理完后返回一个char指针,那么做一个表达式计算器就可能。
#16
vc项目右键设置多字节的。
#17
vc项目右键设置多字节的。
感谢楼上帮了大忙,真心谢谢你。现献图,以供同样问题的新手参考。
在制作DLL的C++项目中按上图设置,(如果用C的话需要修改预编译头和编译为两项),然后在C#调用的时候做以下映射和设置:
[DllImport("Cal_Dll.dll", EntryPoint = "my_strcat", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Ansi)]
public static extern string my_strcat(string a, string b);
textBox3.Text = my_strcat(textBox1.Text, textBox2.Text);
非常感谢!结贴。
#18
http://www.cnblogs.com/Purple_Xiapei/archive/2012/06/28/2568597.html
#19
十分感谢楼主,和楼上的各位大神。
我也是想把自己以前写的C++的代码封装为DLL,用C#调用也是遇到字串乱码的问题,搞了1天快疯了才在楼主这找到答案,真的很感激。
我也是想把自己以前写的C++的代码封装为DLL,用C#调用也是遇到字串乱码的问题,搞了1天快疯了才在楼主这找到答案,真的很感激。
#20
#1
后来,我又在论坛上找到第二个版本如下:
结果不是乱码了,但是只是把两个字符串的首字母连接起来了,盼好心人能指点一二,给出代码以供参考,并能指出我错在哪里。谢谢了。
参考资料如下:
http://blog.csdn.net/xinyaping/article/details/7288164
在此特别感谢金辉软件技术公司的唐先生对我的帮助,谢谢所有帮助过我的人。
[DllImport("*.dll", EntryPoint = "my_strcat", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Unicode)]
public static extern IntPtr my_strcat(string a, string b);
IntPtr pStr = my_strcat(textBox1.Text, textBox2.Text);
textBox3.Text = Marshal.PtrToStringAnsi(pStr);
结果不是乱码了,但是只是把两个字符串的首字母连接起来了,盼好心人能指点一二,给出代码以供参考,并能指出我错在哪里。谢谢了。
参考资料如下:
http://blog.csdn.net/xinyaping/article/details/7288164
在此特别感谢金辉软件技术公司的唐先生对我的帮助,谢谢所有帮助过我的人。
#2
顺便再问一下:
在C#中,如果调用 FILE *fopen(“路径/文件名”, “模式”)这个 fopen 库函数?它返回的是一个文件流指针,在C#中如何操作该指针?若是有高手贴出代码,就太感谢了。
在C#中,如果调用 FILE *fopen(“路径/文件名”, “模式”)这个 fopen 库函数?它返回的是一个文件流指针,在C#中如何操作该指针?若是有高手贴出代码,就太感谢了。
#3
应该是CharSet = CharSet.Unicode这个出了问题,一般来说c/c++缺省的字符集是CharSet.ASCII,你用Unicode传递自然会有乱码,
#4
文件流是c库函数才能使用的东西,c#是没法用的,除非你把c库函数封装成dll,但是没人会这么做,
不要想在c#中操作库函数的流,
不要想在c#中操作库函数的流,
顺便再问一下:
在C#中,如果调用 FILE *fopen(“路径/文件名”, “模式”)这个 fopen 库函数?它返回的是一个文件流指针,在C#中如何操作该指针?若是有高手贴出代码,就太感谢了。
#5
我自己实现了 my_strcat(char *dst, char *src)函数
...
你的C实现会缓存溢出,严重情况下会程序崩溃。
#6
这类问题最好不要去鼓励,相反地应该警告程序员“不要随便使用c++的dll”。
实际上,你应该让c++程序员(而不是自己)去提供.net封装的dll,并且经过严格测试其不会产生bug特别是泄露。
实际上,你应该让c++程序员(而不是自己)去提供.net封装的dll,并且经过严格测试其不会产生bug特别是泄露。
#7
应该是CharSet = CharSet.Unicode这个出了问题,一般来说c/c++缺省的字符集是CharSet.ASCII,你用Unicode传递自然会有乱码
我按你的办法使用了其它字符集,调试的时候VS2010提示: 尝试读取或写入受保护的内存。这通常指示其他内存已损坏。
请问还有其它的办法吗?谢谢了,我主要是学纯C开发,C#是我本来就会一些的,而且在未来的工作中,会很少用C#,大多数都是用C的多,我想使用DLL只是想把C的代码复用,要不然的话要用C#重新写一遍。
乱码的问题纠结了我好久,好想找个人帮我调试一下C#的工程文件,我的 QQ:86389981
盼高手能指点迷津。
#8
这类问题最好不要去鼓励,相反地应该警告程序员“不要随便使用c++的dll”。
实际上,你应该让c++程序员(而不是自己)去提供.net封装的dll,并且经过严格测试其不会产生bug特别是泄露。
嗯,内存泄露的问题确实要注意,不过在写代码的时候注意free就可以使用了吧?我主要是想着把这个C的代码复用,在C#中也可以使用该项功能。如果可能,可不可以帮我看看我的代码哪里出了问题呢?谢谢你。
#9
我自己实现了 my_strcat(char *dst, char *src)函数
...
你的C实现会缓存溢出,严重情况下会程序崩溃。
如果可能,可不可以告诉我在哪个地方应该注意?我是新手。。。
我是第一次发贴,我觉得尽可能在贴子说出自己的想法和遇到的问题。
这样才会有高手看到,并指出自己的不对。
#10
楼主的C版本代码中的 my_strcat 函数有个 bug,你自己发现了吗?那就是缓冲区溢出问题。在你的代码中没有判断des参数的缓冲区是否够用就进行了连接。!
至于C#调用的代码,我还没学那么深,暂时不会。。、
至于C#调用的代码,我还没学那么深,暂时不会。。、
#11
// 危险代码!!
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被要求使用的原因。
#12
可用c++cli封装一层在给c#调用即可了
#13
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
//导入刚刚生成的 DLL ,函数入口是 my_strcat ,调用方式是 cdecl ,字符集是 unicode
[DllImport("test.dll", EntryPoint = "my_strcat", CharSet = CharSet.Ansi)]
public static extern string my_strcat(string a, string b);
private void button1_Click(object sender, EventArgs e)
{
//在点击事件中把文本框1中的字符串,文本框2中的字符串连接起来,然后把连接的字符串放在文本框3中。
textBox3.Text = my_strcat(textBox1.Text, textBox2.Text);
}
}
}
extern "C"
{
__declspec(dllexport) char *my_strcat(char *dst, char *src)
{
char *old_pos = dst;
while (*dst)
{
dst++;
}
while (*dst++ = *src++)
{
;
}
return old_pos;
}
}
vs2008调用成功。
#14
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
//导入刚刚生成的 DLL ,函数入口是 my_strcat ,调用方式是 cdecl ,字符集是 unicode
[DllImport("test.dll", EntryPoint = "my_strcat", CharSet = CharSet.Ansi)]
public static extern string my_strcat(string a, string b);
private void button1_Click(object sender, EventArgs e)
{
//在点击事件中把文本框1中的字符串,文本框2中的字符串连接起来,然后把连接的字符串放在文本框3中。
textBox3.Text = my_strcat(textBox1.Text, textBox2.Text);
}
}
}
extern "C"
{
__declspec(dllexport) char *my_strcat(char *dst, char *src)
{
char *old_pos = dst;
while (*dst)
{
dst++;
}
while (*dst++ = *src++)
{
;
}
return old_pos;
}
}
vs2008调用成功。
呵呵,感谢你的回复,我在论坛其它的贴子中看到你回复乱码的问题。
我分别用 VS2010 VS2008 VS2012 三种版本都测试过你的代码
均没有通过,是否哪个地方需要设置?比如工程属性?
我的 QQ 86389981 可以把你的工程文件传给我吗?
#15
dest 的空间大小需要调用方来保证,像strcpy / strncpy 也是一样的,strcpy 会强行把字符串末尾加个尾零符,这样会改变字符串的,所以都建议用 strncpy, 其实可以使用malloc动态分配内存,这都是后话了。
C的代码在C#里面复用,我纠结了很久,也没有搞明白,为什么是乱码?到底是哪里出了问题。我本来是想做一个计算器(在控制台中测试已实现),输入表达式字符串,然后根据括号和运算符优先级,计算出结果,想象中用C#做界面,用C在后台去处理字符串和算法。
C中没有string类,处理字符串都是指针或者是数组,如何把字符串在C#和C之间传递成了重中之重。如果把表达式的字符串全部用C#的函数处理后计算,就失去了我本意了。虽然不鼓励这样做,但是我只是做个试验,这样做能不能成功?所以就写了这个strcat函数来测试,能不能返回指针,并且形参也是指针。
如果能接收一个char指针,处理完后返回一个char指针,那么做一个表达式计算器就可能。
C的代码在C#里面复用,我纠结了很久,也没有搞明白,为什么是乱码?到底是哪里出了问题。我本来是想做一个计算器(在控制台中测试已实现),输入表达式字符串,然后根据括号和运算符优先级,计算出结果,想象中用C#做界面,用C在后台去处理字符串和算法。
C中没有string类,处理字符串都是指针或者是数组,如何把字符串在C#和C之间传递成了重中之重。如果把表达式的字符串全部用C#的函数处理后计算,就失去了我本意了。虽然不鼓励这样做,但是我只是做个试验,这样做能不能成功?所以就写了这个strcat函数来测试,能不能返回指针,并且形参也是指针。
如果能接收一个char指针,处理完后返回一个char指针,那么做一个表达式计算器就可能。
#16
vc项目右键设置多字节的。
#17
vc项目右键设置多字节的。
感谢楼上帮了大忙,真心谢谢你。现献图,以供同样问题的新手参考。
在制作DLL的C++项目中按上图设置,(如果用C的话需要修改预编译头和编译为两项),然后在C#调用的时候做以下映射和设置:
[DllImport("Cal_Dll.dll", EntryPoint = "my_strcat", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Ansi)]
public static extern string my_strcat(string a, string b);
textBox3.Text = my_strcat(textBox1.Text, textBox2.Text);
非常感谢!结贴。
#18
http://www.cnblogs.com/Purple_Xiapei/archive/2012/06/28/2568597.html
#19
十分感谢楼主,和楼上的各位大神。
我也是想把自己以前写的C++的代码封装为DLL,用C#调用也是遇到字串乱码的问题,搞了1天快疯了才在楼主这找到答案,真的很感激。
我也是想把自己以前写的C++的代码封装为DLL,用C#调用也是遇到字串乱码的问题,搞了1天快疯了才在楼主这找到答案,真的很感激。