C# MarshalAs

来源:互联网 发布:apache jmeter 中文版 编辑:程序博客网 时间:2024/05/23 19:33

转载自:http://blog.csdn.net/xiaobai1593/article/details/7025775

参考: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(),即获取对象的非托管大小时,获得的是自己定义的大小;

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

示例:

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

结构的声明:

[csharp] view plain copy
  1. struct Info  
  2. {  
  3.     [MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]  
  4.     public char[] name;  
  5.     [MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]  
  6.     public char[] cipher;  
  7.     [MarshalAs(UnmanagedType.ByValArray, SizeConst = 256)]  
  8.     public char[] signature;  
  9. }  
结构的使用:

[csharp] view plain copy
  1. Info myInfo;  
  2. myInfo.name = name.ToCharArray();  
  3. myInfo.cipher = cipher.ToCharArray();  
  4. myInfo.signature = signature.ToCharArray();  
注意:

[csharp] view plain copy
  1. 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以后,微软会尽量解决掉这个属性。

原创粉丝点击