C# 与C++的数据转换

时间:2021-09-28 18:23:12

一、类型转化

下面重点罗列下常用的类型转化。

C++类型

C#类型

备注说明

Int

Int16、Int32

没有悬念,直接转化

Uint

UInt16、Uint32、int

在程序中,不太清楚是,就可以直接对应为int

Long

Int32

Long相对int就定型了,对应的就是Int32

DWORD(unsigned long)

Uint32

WORD(unsigned short)

Uint16

这是对WORD的认知。

Byte(Unsigned char)

Byte

DECIMAL

Decimal

位数转化

BOOL

bool

char

char

这种没有加指针,比较容易,直接对应入座

Handle(void *)

Intptr

函数中为窗口句柄,c#就是默认的为Intptr

HMODULE

Intptr

同上

HISTANCE

Intptr

同上

Int *、long *

Ref int、ref long

这种整形指针,在程序中是为了引用,所以在c#中对应的是ref,所以在关键词之前加入ref.

Int &.long &

Ref int 、ref long

解释同上,当然在关键词加入 out,也是可以的

Char *(LPSTR、Pchar)、const char *(LPCSTR)、

String

也是为了获取数据,在c#中string在应用中就是引用,所以直接改为string即可。Intptr也可以,不提倡。

Byte *

Ref byte、byte[]

程序byte * 是为了获取类型为byte数据,在C#则用byte数组获取存储数据。String也可以,但是不提倡,关键看看获取的数据,做什么用

GUID

Guid

Char[],byte[],in[]

可对应找指针类型对一个的转化

Char[],byte[],int,这种在c++中应用时,先初始化数组,然后定义一个指针指向数据地址,然后访问,所以在转化是,在对应为char*》string , int[]>int * > intptr

Char **,byte **

Intptr

这种双指针的调用,一般是访问二位数组,所以我们直接处理为Intptr,当然intptr和之前不一样,需要处理,可以看见例子说明。

结构体 * 变量名、结构体 &变量名。

Ref 结构体 变量名、intptr

在结构体引用,或者传入值c++一般是用指针,在c#中,用ref代替。当然intptr也可以,但是不太方便。

通常在只要你选择在win32运行环境中找到相匹配的CLR(公共语言运行库,负责资源管理:内存分配和回收,并保证应用和底层操作系统之间有必要分离)类型,就可应正常工作。当然也有例外:BOOL在c++中发现其实为int型,所以转化为int,而不是bool。

指针参数,在winAPI许多函数中将指针作为一个或者多个参数。指针的作用是存储数据的地址,而不是数据。指针的加入,增加了数据的复杂性,同时增加数据灵活性,实现数据的传入传出,如果只是值类型应用只能是传入数据。在应用中如果没有指针,您可以直接通过值在线程堆栈中传递数据。有了指针,可以通过引用传递数据,将数据的内存地址推入到线程堆栈中,然后函数通过内存地址间接访问数据。在c#用ref、out定义为类似指针作用关键词。out是ref一个参数规范,实际上他们在运行中产生相同的机器码,out作用为了让调用者明白,数据只是传出,ref表明数据传入也是获取。托管代码中ref、out参数另一个很好用的是,可以作为结构体、类、数组提供一个地址供调用。只有在发现ref或out参数不符合需要情况下,才会封装成更复杂的CLR类型。

在windows API中会有窗口句柄的获取或者赋值,其方法的传递是不透明的,如handle、void *、histance等。

少数情况下,API 函数也将不透明指针定义为 PVOID 或 LPVOID 类型。在 Windows API 的定义中,这些类型意思就是说该指针没有类型。当 一个不透明指针返回给您的应用程序(或者您的应用程序期望得到一个不透明指针)时,您应该将参数或返回值封送为 CLR 中的一种特殊类型 — System.IntPtr。当您使用 IntPtr 类型时,通常不使用 out 或 ref 参数,因为 IntPtr 意为直接持有指针。

在CLR类型系统中intptr是一种特殊的属性,没有固定的大小,在运行时再绑定,依据操作系统的正常指针而定。这意味着在 32 位的 Windows 中,IntPtr 变量的宽度是 32 位的,而在 64 位的 Windows 中,实时编译器编译的代码会将 IntPtr 值看作 64 位的值。当在托管代码和非托管代码之间封送不透明指针时,这种自动调节大小的特点十分有用。然而,当使用 Windows API 函数时,因为指针应是不透明的,所以除了存储和传递给外部方法外,不能将它们另做它用。这种“只限存储和传递”规则的两个特例是当您需要向外部方法传递 null 指针值和需要比较 IntPtr 值与 null 值的情况。为了做到这一点,您不能将零强制转换为 System.IntPtr,而应该在 IntPtr 类型上使用 Int32.Zero 静态公共字段,以便获得用于比较或赋值的 null 值。

封送文本,主要是指在获取数据时,数据可能是存储在char数组中,如果我们用string接收时,有可能为乱码。所以在函数调用过程中,当char *,char[]是作为输入数据时,可以改为string。当作为数据传出时,则要好好考虑了,有时需要改为char []。在c+程序中,就是在c中字符串实际上是只是一个字符值数组,通常为null,大多数windows API函数是按照对于ansi,将其作为字符值数组(比较常用),对于unicode,将其作为宽字符值数组。有时获取的数据为乱码时,可能就是需要转为unicode,就解决了。大多数windows API函数都带有LPTSTR或者LPCTSTR值。他们分别是可修改和不可修改的缓冲区,包含以null结束的字符数组。“C”代表const,意味数据不会传递到函数外部。“T”代表该参数可以是Unicode和ANSI,在CLR运行中取决你选择的字符集和底层操作系统的字符集.。所以函数声明时,加上DllImportAttribute 为CharSet.Auto就可以了。如果字符串参数只用作输入,则使用 System.String 类型。在托管代码中,字符串是不变的,适合用于不会被本机 API 函数更改的缓冲区。如 果字符串参数可以用作输入和/或输出,则使用 System.StringBuilder 类型。StringBuilder 类型是一个很有用的类库类型,它可以帮助您有效地构建字符串,也正好可以将缓冲区传递给本机函数,由本机函数为您填充字符串数据。一旦函数调用返回,您只 需要调用 StringBuilder 对象的 ToString 就可以得到一个 String 对象。

CharSet的各变量对char以及char[]的影响如下:
ANSI:char以及char[]占一个字节
AUTO:char以及char[]占两个字节
UNICODE:char以及char[]占两个字节

总体原则可以总结为:

1、在c++常用的基本类型(数值类型、字节类型)直接转化到c#中的数值类型。(原则是字节数,确定好)

2、在c++常用的指针类型(数值类型*、字节类型*)则转化为c#中的ref 数值类型、ref 字节类型。但是在常用时,针对char * ,转化为string。

3、在c++常用的构造类型(结构体、数组、枚举类型、共用体)行对比较复杂。枚举和共用体直接复制就可以用,结构体的声明随后重点讲,在函数调用时,如果为结构体 * 变量名,则为 ref 结构体 变量名。数组在 函数调用,可以直接写为数组名,也可以写为Intptr。

二、结构体

1、结构体的重定义。

在c++中会有很多结构体,结构体内有各种各样的数据类型,所以就牵涉到数据类型的转化,同时在通过结构体获取到数据后,也牵涉到编码转化问题。

结构体类型和类类型在语法上有很多相似之处,他们都是一种数据结构,都可以包括数据成员和方法成员。

结构体和类区别:

1、结构体是值类型,它在栈中分配空间;类是引用类型,他在堆中分配空间,栈存储的是引用。

2、结构体类型直接存储成员数据,类中数据类型存在堆中,然后通过在栈中引用,访问数据。因为结构体是值类型,直接存储,因此当对象的主要成员为数据切不大时,使用结构体效率更高。

3、结构体直接包含自己的数据,每个结构体都保存一份数据,在程序声明两个结构体对象,改变其中一个,另一个数据不变,但是类是引用,则另一个数据会改变。

4、结构体是值类型,不能初始化为null,复制时则数据全部复制。;类中的复制是引用的复制,数据较大时,结构体复制则效果不是很好。

结构和类的适用场合分析:

  1、当堆栈的空间很有限,且有大量的逻辑对象时,创建类要比创建结构好一些;

  2、对于点、矩形和颜色这样的轻量对象,假如要声明一个含有许多个颜色对象的数组,则CLR需要为每个对象分配内存,在这种情况下,使用结构的成本较低;

  3、在表现抽象和多级别的对象层次时,类是最好的选择,因为结构不支持继承。

  4、大多数情况下,目标类型只是含有一些数据,或者以数据为主。

结构体的声明、初始化、引用在c#还是很重要的,下面根据代码进行分析。

  1. //C++的结构体
  1. //录像索引列表文件
  2. typedef struct tagINDEX_INFO
  3. {
  4. DWORD dwStartTime;              //录像开始时间
  5. DWORD dwEndTime;                //录像停止时间
  6. BYTE  btFileType;                   //文件类型
  7. BYTE  btFileStatus;             //文件状态
  8. BYTE  Reserved[2];              //预留,LMC向NVR请求回放时Reserved[0]标识NVR发送速度,Reserved[1]存放录像倒放标志
  9. BYTE  btMAC[6];                 //设备MAC地址
  10. WORD  wChan;                    //设备通道
  11. DWORD dwIP1;                    //设备IP1
  12. DWORD dwIP2;                    //设备IP2,公网模式下,存储此文件所在的NVR的IP
  13. DWORD dwIP3;                    //设备IP3,V3061用来存储录像片段在录像文件的偏移量,勿动
  14. DWORD dwIP4;                    //设备IP4,V3061用来录像文件名,勿动
  15. DWORD dwFileOffset;             //文件偏移,用于文件下载时断点续传和定位
  16. DWORD dwReserved;               //预留,3070有用到,不要动它
  17. }INDEX_INFO, *LPINDEX_INFO;

C#结构体

//录像索引列表文件
        [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi,Pack = 1)]
        public struct NET_INDEX_INFO
        {
            public UInt32 dwStartTime; //录像开始时间
            public UInt32 dwEndTime; //录像停止时间
            public byte btFileType; //文件类型
            public byte btFileStatus; //文件状态
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 2)]
            public byte [] Reserved; //预留,LMC向NVR请求回放时Reserved[0]标识NVR发送速度,Reserved[1]存放录像倒放标志
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]
            public byte [] btMAC; //设备MAC地址
            public UInt16 wChan; //设备通道
            public UInt32 dwIP1; //设备IP1
            public UInt32 dwIP2; //设备IP2,公网模式下,存储此文件所在的NVR的IP
            public UInt32 dwIP3; //设备IP3,V3061用来存储录像片段在录像文件的偏移量,勿动
            public UInt32 dwIP4; //设备IP4,V3061用来录像文件名,勿动
            public UInt32 dwFileOffset; //文件偏移,用于文件下载时断点续传和定位
            public UInt32 dwReserved; //预留,3070有用到,不要动它
        }

上面成员前面必须添加public,因为默认是private。

2、StructLayout特性

公共语言运行库利用StructLayoutAttribute控制类或结构的数据字段在托管内存中的物理布局,即类或结构需要按某种方式排列。如果要将类传递给需要指定布局的非托管代码,则显式控制类布局是重要的。它的构造函数中用 LayoutKind值初始化 StructLayoutAttribute 类的新实例。 LayoutKind.Sequential 用于强制将成员按其出现的顺序进行顺序布局。
System.Runtime.InteropServices.StructLayout   允许的值有StructLayout.Auto   StructLayout.Sequential   StructLayout.Explicit.  在应用中为了数据顺利传入,一般是用StructLayout.Sequential,意味着结构体体内数据传入的格局按照声明一样。
StructLayout.Explicit需要用FieldOffset()设置每个成员的位置这样就可以实现类似c的公用体的功能。
[StructLayout(LayoutKind.Explicit)] 
struct S1
{
  [FieldOffset(0)]
  int a;
  [FieldOffset(0)]
  int b;
}
这样a和b在内存中地址相同

StructLayout特性支持三种附加字段:CharSet、Pack、Size。     
·   CharSet定义在结构中的字符串成员在结构被传给DLL时的排列方式。可以是Unicode、Ansi或Auto。     
  默认为Auto,在WIN   NT/2000/XP中表示字符串按照Unicode字符串进行排列,在WIN   95/98/Me中则表示按照ANSI字符串进行排列。     
·   Pack定义了结构的封装大小。可以是1、2、4、8、16、32、64、128或特殊值0。特殊值0表示当前操作平台默认的压缩大小。

3.MarshalAs的使用

MarshalAs属性指示如何在托管代码和非托管代码之间封送数据。
[MarshalAs(UnmanagedType unmanagedType, 命名参数)]

常用的UnmanagedType枚举值:(详细内容查MSDN)

BStr   长度前缀为双字节的 Unicode 字符串;

LPStr  单字节、空终止的 ANSI 字符串。;

LPWStr  一个 2 字节、空终止的 Unicode 字符串;

ByValArray 用于在结构中出现的内联定长字符数组,应始终使用MarshalAsAttribute的SizeConst字段来指示数组的大小。常用的是数组对应的为ByAvlArray,string对应的是ByValStr。

C# 与C++的数据转换的更多相关文章

  1. 利用Python进行数据分析(14) pandas基础: 数据转换

    数据转换指的是对数据的过滤.清理以及其他的转换操作. 移除重复数据 DataFrame里经常会出现重复行,DataFrame提供一个duplicated()方法检测各行是否重复,另一个drop_dup ...

  2. Linux C编程学习6---字符串处理、数据转换

    1.字符串 应用程序按其功能可分为数值计算.非数值计算以及输入输出操作等.非数值计算程序占相当大的比例,其核心就是字符串处理1.1.字符测试 1.1.1.测试字符是否为英文字母 int isalpha ...

  3. .NET LINQ数据转换

    使用 LINQ 进行数据转换      语言集成查询 (LINQ) 不仅可用于检索数据, 而且还是一个功能强大的数据转换工具. 通过使用 LINQ 查询,您可以将源序列用作输入,并采用多种方式修改它以 ...

  4. 【转】C#中将JSon数据转换成实体类,将实体类转换成Json

    http://wo13145219.iteye.com/blog/2022667 http://json2csharp.chahuo.com/ using System; using System.C ...

  5. LINQ之路 7:子查询、创建策略和数据转换

    在前面的系列中,我们已经讨论了LINQ简单查询的大部分特性,了解了LINQ的支持计术和语法形式.至此,我们应该可以创建出大部分相对简单的LINQ查询.在本篇中,除了对前面的知识做个简单的总结,还会介绍 ...

  6. SSIS数据转换后数值总数差异过大

    之前做过一个项目,犯了一个小错误,写出来给大家分享一下,以防大家出同样的错误. 做了一个ETL包,对货品的销售额进行数据转换,字符型 --〉 浮点型: 之后对销售额进行求和,在测试数据结果时发现与销售 ...

  7. SpringMVC 数据转换 & 数据格式化 & 数据校验

    数据绑定流程 1. Spring MVC 主框架将 ServletRequest 对象及目标方法的入参实例传递给 WebDataBinderFactory 实例,以创建 DataBinder 实例对象 ...

  8. 关于JAVA的数据转换总结

    数据转换在编程里面是十分常用的,将平常可能用到的数据转换类型总结起来会在以后码代码的过程中有很大帮助. 在数据转换之前,需要明白的是基础数据类型的自动转换和强制转换.接下来就先从数据类型的容量讲起. ...

  9. mysql的数据转换

    在sql语句中完成对数字类型的数据转换成字符类型的数据.像这次将读取出来的float类型的数据,在进行jsonObject.fromObject(object).toString();这个方法,并没有 ...

  10. Json数据与Json数据转换

    1.json数据 [{\"IS_DISTRIBUTOR_LIMIT\":0,\"PROVISION_PRICE\":null,\"PRO_STATUS ...

随机推荐

  1. Angularjs ng-if和ng-show的区别

    ng-if:判断条件,为true时向html中插入节点,否则从html中移除节点: ng-show: 简单的显示和隐藏(display) 坑点:ng-if会创建一个新的作用域(scope),如果内部元 ...

  2. 白话贝叶斯理论及在足球比赛结果预测中的应用和C#实现

    离去年“马尔可夫链进行彩票预测”已经一年了,同时我也计划了一个彩票数据框架的搭建,分析和预测的框架,会在今年逐步发表,拟定了一个目录,大家有什么样的意见和和问题,可以看看,留言我会在后面的文章中逐步改 ...

  3. MongoDB 权限

    1.使用mongod 启动后(加入了--auth后操作数据库则需要权限) mongod --dbpath=D:\mongdb\db --logpath=D:\mongodb\log.txt --por ...

  4. Least Common Multiple

    地址:http://www.codewars.com/kata/5259acb16021e9d8a60010af/train/python 题目: Write a function that calc ...

  5. 怎样制作PHP验证码?

    <?php /** *制作验证码 *1.启动session *2.设定标头 *3.创建画布 *4.创建颜色 *5.创建随机数并放到画布上 *6.将得到的若干随机数放入session中 *7.添加 ...

  6. Cannot create an instance of OLE DB provider &OpenCurlyDoubleQuote;OraOLEDB&period;Oracle” for linked server &quot&semi;xxxxxxx&quot&semi;&period;

    在SQL SERVER 2008 R2下用Windows 身份认证的登录名创建了一个访问ORACLE数据库的链接服务器xxxxx,测试成功,木有问题,但是其它登录名使用该链接服务器时,报如下错误: 消 ...

  7. Zend Studio下的PHP代码调试

    问题:Zend Studio无法调试php代码 安装Zend Debugger 下载 到http://downloads.zend.com/pdt/server-debugger下载最新的debugg ...

  8. Android各版本代号、版本号、API&sol;NDK级别、发布时间及市场份额

    Android各版本代号.版本号.API/NDK级别.发布时间及市场份额 代号 版本号 API/NDK级别 发布时间 - O 8.0 API level 26 2017-3-21 牛轧糖 Nougat ...

  9. Innodb表空间

    Innodb有两种管理表空间的方法 独立表空间:每一张表都会生成独立的文件来进行存储,每一张表都有一个.frm表描述文件,和一个.ibd文件.其中ibd文件包括了单独一个表的数据内容和索引内容. 共享 ...

  10. Maven项目mybatis Invalid bound statement &lpar;not found&rpar;解决方法

    最近因为工作需要,要学习mybatis框架.在添加好一些依赖之后,通过mybatis进行数据库的crud操作.但是在测试的时候总是报mybatis:Invalid bound statement (n ...