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

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,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,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,pt,1); 
 
 int count = HSCAN_ReadCANMessage(0x0,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(); 
 } 

以上这篇基于C#调用c++Dll结构体数组指针的问题详解就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持编程小技巧。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


1 在Visual Studio 中创建一个Asp.NET WebApi 项目,项目名:Com.App.SysApi(本例创建的是 .net 4.5 框架程序) 2 打开Nuget 包管理软件,查找 swagger 安装Swashbuckle 组件 3 开启项目的XML注释位置(选中项目Com.Ap
using ImpromptuInterface; using System; using System.Dynamic; namespace ConsoleApp2 { class Program { static void Main(string[] args) { dynamic expand
https://blog.csdn.net/michaelgong/article/details/43148597
C#11添加了文件作用域类型功能:一个新的file修饰符,可以应用于任何类型定义以限制其只能在当前文件中使用。 这样,我们可以在一个项目中拥有多个同名的类。 通过下面的项目显示,该项目包含两个名为Answer的类。 文件File1.cs中 namespace ConsoleApp11 { file
在.NET中Newtonsoft.Json(Json.NET)是我们常用来进行Json序列化与反序列化的库。 而在使用中常会遇到反序列化Json时,遇到不规则的Json数据解构而抛出异常。 Newtonsoft.Json&#160;支持序列化和反序列化过程中的错误处理。 允许您捕获错误并选择是处理它
设置允许跨域访问的网址--&gt; 设置预检有效时间--&gt; --&gt;
先来看一点实例,两个类之间的映射。 首先定义两个类Source与DTOSource: Source与DTOSource字段完全相同,来看看它俩如何通过AutoMapper转换,代码很简单。 第一步建立Source到DTOSource之间的映射,初始化一个Source实例后,来看下执行结果: 执行完成
C#中提供了IObservable&lt;T&gt;接口和IObserver&lt;T&gt;接口来实现观察者模式,IObservable&lt;T&gt;相当于Subject(主题)接口,下面我们就以代吗来说明下如何利用.net框架提供的观察者模式接口。 WeatherData类包含气温,湿度,气
class Program { private static readonly DiagnosticSource testDiagnosticListener = new DiagnosticListener(&quot;TestDiagnosticListener&quot;); public s
.Net Core在调用其他服务时,调用通常使用HttpClient,而HttpClient默认使用HTTP/1.1 。 配置 HttpClient 以使用 HTTP/2 h2 连接 自 .NET Core 3.0 发布以来, .NET 开发人员可以使用 HttpClient 启用 HTTP/2 。
转自:https://www.cnblogs.com/neverc/p/5241466.html AOP介绍 面向切面编程(Aspect Oriented Programming,英文缩写为AOP),通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。 AOP是OOP的延续,是软件开发中
&lt;PropertyGroup&gt; &lt;RunWorkingDirectory&gt;../../&lt;/RunWorkingDirectory&gt; &lt;/PropertyGroup&gt;
https://www.cnblogs.com/ideacore/p/7803606.html 1.安装T4 Toolbox 2.新建T4 Toolbox模板 3.编辑T4 Toolbox模板&#160;Bbing_Template_Mongo_Service.tt 4.添加T4模板 Bbing_T
using System; using System.Collections.Generic; using System.Text; using Dapper; using System.Data; using System.Data.SqlClient; using System.Linq; us
using cpf360.Common; using cpf360.ModelInfo; using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Thread
public ActionResult Login(string data) { var _params = JsonConvert.DeserializeAnonymousType(data, new { userName = &quot;&quot;, password = &quot;&quo
什么是Elastic AMP Elastic APM 是一个应用程序性能监控系统。它可以请求的响应时间、数据库查询、对缓存的调用、外部 HTTP 请求等的详细性能信息,可以实时监控软件服务和应用程序。这可以帮助我们快速查明和修复性能问题。 Elastic APM 还会自动收集未处理的错误和异常。因此
#user nobody; worker_processes 1; #error_log logs/error.log; #error_log logs/error.log notice; #error_log logs/error.log info; #pid logs/nginx.pid; ev
背景 前些天遇到一个需求,在没有第三方源码的情况下,刷新一个第三方UI,并且拦截到其ajax请求的返回结果。当结果为AVALIABLE的时候,停止刷新并语音提示,否则继续刷新。 分析这个需求,发现需要控制一个刷新循环的暂停与开始,因此网上搜到了通过ManualResetEvent实现线程的暂停与恢复
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; using cpf360.Common; using cpf360.DTO; name