基于C#调用c++Dll结构体数组指针的问题详解

作者:jadeflute 时间:2021-12-10 23:16:41 

C#调用c++dll文件是一件很麻烦的事情,首先面临的是数据类型转换的问题,相信经常做c#开发的都和我一样把学校的那点c++底子都忘光了吧(语言特性类)。

网上有一大堆得转换对应表,也有一大堆的转换实例,但是都没有强调一个更重要的问题,就是c#数据类型和c++数据类型占内存长度的对应关系。

如果dll文件中只包含一些基础类型,那这个问题可能可以被忽略,但是如果是组合类型(这个叫法也许不妥),如结构体、类类型等,在其中的成员变量的长度的申明正确与否将决定你对dll文件调用的成败。

如有以下代码,其实不是dll文件的源码,而是厂商给的c++例子代码

c++中的结构体申明


typedef struct
{
unsigned char Port;
unsigned long Id;
unsigned char Ctrl;
unsigned char pData[8];
}HSCAN_MSG;

c++中的函数申明(一个c++程序引用另一个c++的dll文件)


extern "C" int _stdcall HSCAN_SendCANMessage(unsigned char nDevice,unsigned char nPort,HSCAN_MSG *msg,int nLength);

c++中的调用:


....
HSCAN_MSG msg[100];
.....
HSCAN_SendCANMessage(m_nDevice,m_nPort,msg,nFrames);

由上述代码可见,msg是个结构体的数组。

下面是我的c#的代码

c#结构体申明:(申明成)


[StructLayout(LayoutKind.Sequential)]
public struct HSCAN_MSG
{
// UnmanagedType.ByValArray, [MarshalAs(UnmanagedType.U1)]这个非常重要,就是申明对应类型和长度的
[MarshalAs(UnmanagedType.U1)]
public byte Port;
[MarshalAs(UnmanagedType.U4)]
public uint nId;
[MarshalAs(UnmanagedType.U1)]
public byte nCtrl;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)]
public byte[] pData;
};

c#函数申明


[DllImport("HS2106API.dll")]
public static extern int HSCAN_SendCANMessage(
byte nDevice, byte nPort, HSCAN_MSG[] pMsg, int nLength);

C#函数调用


HSCAN_MSG[] msg = new HSCAN_MSG[1]; //发送缓冲区大小可根据需要设置;
for (int yy = 0; yy < msg.Length; yy++)
{
msg[yy] = new HSCAN_MSG();
}
//...结构体中的成员的实例化略
HSCAN_SendCANMessage(0x0, 0x0, msg, 1)

那些只能用指针不能用结构体和类的地方

c++中的结构体申明


typedef struct
{
unsigned char Port;
unsigned long Id;
unsigned char Ctrl;
unsigned char pData[8];
}HSCAN_MSG;

c++中的函数申明(一个c++程序引用另一个c++的dll文件)


extern "C" int _stdcall HSCAN_SendCANMessage(unsigned char nDevice,unsigned char nPort,HSCAN_MSG *msg,int nLength);

c#中的结构体申明:


[StructLayout(LayoutKind.Sequential)]
public struct HSCAN_MSG
{
[MarshalAs(UnmanagedType.U1)]
public byte Port;
/// <summary>
/// 节点标识,nEFF=1 时(扩展帧),为29 位nEFF=0(标准帧)时,为11 位;
/// </summary>
[MarshalAs(UnmanagedType.U4)]
public uint nId;
[MarshalAs(UnmanagedType.U1)]
public byte nCtrl;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)]
public byte[] pData;
};

c#函数的调用:包含使用指针IntPtr替代结构体数组和读取IntPtr的方法


HSCAN_MSG[] msg1 = new HSCAN_MSG[10];
for (int i = 0; i < msg1.Length; i++)
{
msg1[i] = new HSCAN_MSG();
msg1[i].pData = new byte[8];
}
IntPtr[] ptArray = new IntPtr[1];
ptArray[0] = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(HSCAN_MSG)) * 10);
IntPtr pt = Marshal.AllocHGlobal(Marshal.SizeOf(typeof(HSCAN_MSG)));
Marshal.Copy(ptArray, 0, pt, 1);

int count = HSCAN_ReadCANMessage(0x0, 0,pt, 10);

textBoxStatus.Text += "/r/n" + "读取0口:" + count.ToString() + "帧数据";
for (int j = 0; j < 10; j++)
{
msg1[j] =
(HSCAN_MSG)Marshal.PtrToStructure((IntPtr)((UInt32)pt+ j * Marshal.SizeOf(typeof(HSCAN_MSG)))
, typeof(HSCAN_MSG));
textBoxStatus.Text += "/r/n收到0口" + Convert.ToByte(msg1[j].pData[0]).ToString()
+ "|" + Convert.ToByte(msg1[j].pData[1]).ToString()
+ "|" + Convert.ToByte(msg1[j].pData[2]).ToString()
+ "|" + Convert.ToByte(msg1[j].pData[3]).ToString()
+ "|" + Convert.ToByte(msg1[j].pData[4]).ToString()
+ "|" + Convert.ToByte(msg1[j].pData[5]).ToString()
+ "|" + Convert.ToByte(msg1[j].pData[6]).ToString()
+ "|" + Convert.ToByte(msg1[j].pData[7]).ToString();
}

来源:http://blog.csdn.net/wumuzhizi/article/details/48180167

标签:C#,调用,c++,Dll,结构体,数组,指针
0
投稿

猜你喜欢

  • C#根据http和ftp图片地址获取对应图片

    2023-06-06 02:00:23
  • 详解SpringCloud服务认证(JWT)

    2023-11-14 11:19:43
  • 基于Java随机生成手机短信验证码的实例代码

    2023-12-22 02:50:49
  • Java工具类DateUtils实例详解

    2022-08-22 00:40:35
  • java读写ini文件、FileOutputStream问题

    2023-11-29 08:50:13
  • C#实现将DataTable内容输出到Excel表格的方法

    2023-05-01 15:20:38
  • flutter material widget组件之信息展示组件使用详解

    2023-06-22 08:45:35
  • 浅谈java+内存分配及变量存储位置的区别

    2022-07-09 00:46:47
  • Java设计模式之享元模式实例详解

    2021-12-19 17:54:00
  • MyBatis中的properties配置(推荐)

    2023-06-04 20:47:58
  • Java输入年份和月份判断多少天实例代码

    2023-12-23 10:43:11
  • IDEA 2019.2.3破解激活教程(亲测有效)

    2023-02-21 22:16:23
  • Spring Boot和Kotlin的无缝整合与完美交融

    2022-07-08 04:54:09
  • java 读取本地文件实例详解

    2023-08-12 20:41:32
  • IDEA GIT 忽略文件的最佳方式推荐

    2023-04-06 10:02:54
  • spring MVC实践需要注意的地方

    2023-02-07 00:33:07
  • 详解Android的.aar文件生成方法以及使用技巧

    2023-08-06 06:43:20
  • java 交换两个数据的方法实例详解

    2021-12-06 00:56:04
  • 线程阻塞唤醒工具 LockSupport使用详解

    2023-11-29 17:16:10
  • Java MongoDB数据库连接方法梳理

    2023-11-25 01:01:20
  • asp之家 软件编程 m.aspxhome.com