C#与C++结构体的交互

C#在和C++进行交互时,有时候会需要传递结构体。

做一些总结,避免大家在用的时候踩坑。

一般情况

例如我们在C++里定义了一个struct_basic结构体

复制代码
1 struct struct_basic
2 {
3     WORD  value_1;
4     LONG  value_2;
5     DWORD value_3;
6     UINT  value_4;
7     BOOL  value_5;
8 };

这种情况在C#里调用,只需要注意数据类型对应就行了

在C#里定义如下

复制代码
1  struct struct_basic
2     {
3         public ushort value_1;
4         public int value_2;
5         public uint value_3;
6         public uint value_4;
7         public bool value_5;
8     }

使用导出函数来测试一下

复制代码
 1 extern "C" __declspec(dllexport) void get_basic(struct_basic basic);
 2 
 3 extern "C" __declspec(dllexport) void get_basic(struct_basic basic)
 4 {
 5     basic.value_1 = 1;
 6     basic.value_2 = 2;
 7     basic.value_3 = 3;
 8     basic.value_4 = 4;
 9     basic.value_5 = FALSE;
10 }

在C#中调用

复制代码
        [DllImport("lib.dll")]
        public static extern void get_basic(ref struct_basic basic);


        static void Main(string[] args)
        {
            struct_basic basic = new struct_basic();
            get_basic(ref basic);
            Console.WriteLine($"{basic.value_1}  {basic.value_2}  {basic.value_3}  {basic.value_4}  {basic.value_5}");
        }

输出结果如下:

包含字符串的情况

C++

复制代码
1 struct struct_advanced
2 {
3     WORD id;
4     CHAR message[256];
5 };

这种需要使用MarshalAsAttribute特性来指示如何在托管代码与非托管代码之间封送数据。

复制代码
UnmanagedType.ByValTStr表示 结构中大小固定的字符串
复制代码
SizeConst可以指定字符数组的大小

说明:
如果未指定CharSet,默认使用的是Ansi(char)。
Unicode为w_char。

C#

复制代码
1     [StructLayout(LayoutKind.Sequential,CharSet=CharSet.Unicode)]
2     struct struct_advanced
3     {
4         public ushort id;
5 
6         [MarshalAs(UnmanagedType.ByValTStr,SizeConst =256)]
7         public string message;
8     };

使用导出函数测试

C++

复制代码
1 extern "C" __declspec(dllexport) void get_advanced(struct_advanced* advanced);
2 
3 extern "C" __declspec(dllexport) void get_advanced(struct_advanced* advanced)
4 {
5     advanced->id = 101;
6     lstrcpyW(advanced->message, L"HelloWorld");
7 }

在C#中调用

复制代码
 1  class Program
 2     {
 3         [DllImport("lib.dll")]
 4         public static extern void get_advanced(ref struct_advanced advanced);
 5 
 6         static void Main(string[] args)
 7         {
 8             struct_advanced advanced = new struct_advanced();
 9             get_advanced(ref advanced);
10             Console.WriteLine($"{advanced.id}  {advanced.message}");
11         }
12     }

输出结果:

示例代码

相关推荐
D_evil__几秒前
[C++高频精进] 并发编程:线程基础
c++
Mr_WangAndy39 分钟前
C++17 新特性_第二章 C++17 语言特性_std::any和string_view
c++·string_view·c++40周年·c++17新特性·c++新特性any
水天需0102 小时前
C++ 三种指针转换深度解析
c++
言言的底层世界3 小时前
c++中STL容器及算法等
开发语言·c++·经验分享·笔记
Mr_WangAndy3 小时前
C++17 新特性_第一章 C++17 语言特性___has_include,u8字符字面量
c++·c++40周年·c++17新特性·__has_include·u8字面量
liu****3 小时前
八.函数递归
c语言·开发语言·数据结构·c++·算法
Vanranrr4 小时前
C++临时对象与悬空指针:一个导致资源加载失败的隐藏陷阱
服务器·c++·算法
BestOrNothing_20154 小时前
【C++基础】Day 5:struct 与 class
c++·c·class类·struct结构体·typename模板·private与public
枫叶丹44 小时前
【Qt开发】Qt窗口(三) -> QStatusBar状态栏
c语言·开发语言·数据库·c++·qt·microsoft
Skrrapper4 小时前
【编程史】微软的起家之路:一代传奇的诞生
数据库·c++·microsoft