zoukankan      html  css  js  c++  java
  • C# 与C++的数据转换

    在集成工作中,经常会有用c#代码调用c++的dll,这里难免会有类型转化。在调用中经常出现的问题有;

    一、类型转化

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

    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#还是很重要的,下面根据代码进行分析。

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

    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。


  • 相关阅读:
    【系列】CentOS 7.3 离线安装(无网络环境)CI CD环境之sonarqube配置
    Abp vnext 配置Swagger增加token认证
    sonarqube+gitlab runner +docker 代码质量检查问题汇总
    【EF Core】EF core中使用FluentAPI对外键进行指定配置
    【系列】CentOS 7.3 离线安装(无网络环境)CI CD环境之gitlab runner 关于私有docker仓库配置
    【系列】CentOS 7.3 离线安装(无网络环境)CI CD环境之harbor
    【系列】CentOS 7.3 离线安装(无网络环境)CI CD环境之gitlab + gitlab runner(docker in docker)
    【杂记】关于在实际项目中使用TDD的方法
    【系列】CentOS 7.3 离线安装(无网络环境)CI CD环境之docker+docker compose
    【TeamCity】使用TeamCity搭建ASP.NET Core + SVN 的 CICD环境
  • 原文地址:https://www.cnblogs.com/polly333/p/4498402.html
Copyright © 2011-2022 走看看