MarshalAs的使用

来源:互联网 发布:圣诞树 知乎 编辑:程序博客网 时间:2024/05/23 18:38

参考:http://blog.sina.com.cn/s/blog_4e4ee8ed0100elou.html

作用:

MarshalAs属性指示如何在托管代码和非托管代码之间封送数据。

使用方法:

[MarshalAs(UnmanagedType unmanagedType, 命名参数)]

实际上相当于构造一个MarshalAsAttribute类的对象

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

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

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

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

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

注意:

在用Marshal.SizeOf(),即获取对象的非托管大小时,获得的是自己定义的大小;

但在实际处理的时候,是按照实际的大小来获取的

示例:

定义一个固定大小的结构体,代码如下:

结构的声明:

        struct Info        {            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]            public char[] name;            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]            public char[] cipher;            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 256)]            public char[] signature;        }
结构的使用:

            Info myInfo;            myInfo.name = name.ToCharArray();            myInfo.cipher = cipher.ToCharArray();            myInfo.signature = signature.ToCharArray();
注意:

int size = Marshal.SizeOf(myInfo);
size=16+16+256

可见,获取到的非托管大小为288

但是,查看myInfo对象可以看到其实际大小如下所示:


问题:

这种实际大小和固定大小的不一致性,导致了在用Marshal类进行托管对象和非托管对象的转换时,会有如下错误提示:“未能封送类型,因为嵌入数组实例的长度与布局中声明的长度不匹配。”

解决办法还没想到......

小结:
MarshalAs这个属性很难用,很容易用错,用好需要对C#、C++和COM数据的布局方式有一定的了解才能做。所以做好使用一些工具来帮你,可以参照我下面的文章:
http://blog.csdn.net/Donjuan/archive/2009/02/05/3865026.aspx
如果你只是感兴趣的话,那就忘了这个属性吧,在.NET 4.0以后,微软会尽量解决掉这个属性。