二 结构的默认封送处理

  格式化类型是一种复杂类型,它包含显式控制其成员在内存中的布局的信息。成员布局信息是使用 StructLayoutAttribute 属性提供的。布局可以是下列 LayoutKind 枚举值之一:

·                 LayoutKind.Automatic

指示公共语言运行库可以随意重新排列该类型的成员的顺序以提高效率。但是,将值类型传递给非托管代码时,成员的布局是可预知的,类传递的时候默认是这种布局。

·                 LayoutKind.Sequential

指示该类型的成员要在非托管内存中以它们在托管类型定义中出现的同一顺序布局, 结构传递的时候默认布局。

·                 LayoutKind.Explicit

指示成员被根据随每个字段提供的 FieldOffsetAttribute 布局。

                                                  平台调用中使用的值类型

  在下面的示例中,Point Rect 类型使用 StructLayoutAttribute 提供成员布局信息。

using System.Runtime.InteropServices;

[StructLayout(LayoutKind.Sequential)]

public struct Point {

   public int x;

   public int y;

}  

 

[StructLayout(LayoutKind.Explicit)]

public struct Rect {

   [FieldOffset(0)] public int left;

   [FieldOffset(4)] public int top;

   [FieldOffset(8)] public int right;

   [FieldOffset(12)] public int bottom;

}

  当封送到非托管代码时,这些格式化类型作为 C 样式的结构封送。这就提供了一种调用具有结构参数的非托管 API 的简单方法。例如,POINT RECT 结构可以通过下面的方式传递给 Microsoft Win32 API PtInRect 函数:

BOOL PtInRect(const RECT *lprc, POINT pt);

  可以使用下面的平台调用定义传递结构:

class Win32API {

   [DllImport("User32.dll")]

   public static extern Bool PtInRect(ref Rect r, Point p);

}

  Rect 值类型必须通过引用传递,原因是非托管 API 要求将指向 RECT 的指针传递给该函数。而因为非托管 API 要求在堆栈上传递 POINT,因此 Point 值类型通过值传递。引用是作为指针传递给非托管代码的。值在堆栈上传递给非托管代码。

注意

当格式化的类型作为结构封送时,只有该类型内的字段是可访问的。如果该类型具有方法、属性或事件,将无法从非托管代码访问它们。

 

  类还可以作为 C 样式的结构封送到非托管代码,条件是它们具有固定的成员布局。类的成员布局信息也是用 StructLayoutAttribute 属性提供的。具有固定布局的值类型与具有固定布局的类之间的主要差异在于将它们封送到非托管代码的方式。值类型通过值传递(在堆栈上),因此被调用方对类型的成员所做的任何更改对于调用方都是不可见的。引用类型通过引用传递(对类型的引用在堆栈上传递);因此,被调用方对类型的可直接复制到本机结构中的类型成员所做的所有更改对于调用方都是可见的。

注意

如果引用类型具有非直接复制到本机结构中的类型的成员,则需要进行两次转换:第一次在将参数传递到非托管端时进行,而第二次在从调用返回时进行。由于这项附加的开销,如果调用方希望看见被调用方所做的更改,则必须将 In/Out 参数显式应用于参数。

在下面的示例中,SystemTime 类具有连续的成员布局,并且可以传递给 Win32 API GetSystemTime 函数。

[StructLayout(LayoutKind.Sequential)]

   public class SystemTime {

   public ushort wYear;

   public ushort wMonth;

   public ushort wDayOfWeek;

   public ushort wDay;

   public ushort wHour;

   public ushort wMinute;

   public ushort wSecond;

   public ushort wMilliseconds;

}

 

GetSystemTime 函数以下面的方式定义:

void GetSystemTime(SYSTEMTIME* SystemTime);

GetSystemTime 的等效平台调用定义如下所示:

class Win32API {

   [DllImport("Kernel32.dll", CharSet=CharSet.Auto)]

   public static extern void GetSystemTime(SystemTime st);

}

请注意,SystemTime 参数未类型化为引用参数,原因是 SystemTime 是类而不是值类型。与值类型不同,类始终要通过引用来传递。

下面的代码示例显示一个不同的 Point 类,它具有称为 SetXY 的方法。由于该类型具有连续布局,因此可将它传递给非托管代码并作为结构封送。但是,即使通过引用传递对象,仍然不能从非托管代码调用 SetXY 成员。

[StructLayout(LayoutKind.Sequential)]

public class Point {

   int x, y;

   public void SetXY(int x, int y){

      this.x = x;

      this.y = y;

   }

}

COM Interop 中使用的值类型

还可以将格式化类型传递给 COM Interop 方法调用。事实上,当导出到类型库时,值类型被自动转换为结构。如下面的示例所示,Point 值类型变为名为 Point 的类型定义 (typedef)。类型库中其他地方的所有对 Point 值类型的引用都被替换为 Point typedef

类型库表示形式

typedef struct tagPoint {

   int x;

   int y;

} Point;

interface _Graphics {

  

   HRESULT SetPoint ([in] Point p)

   HRESULT SetPointRef ([in,out] Point *p)

   HRESULT GetPoint ([out,retval] Point *p)

}

在通过 COM 接口进行封送处理时,使用的规则与封送值和对平台调用的调用的引用时使用的规则相同。例如,将 Point 值类型的实例从 .NET Framework 传递给 COM 时,Point 通过值传递。如果 Point 值类型通过引用传递,则指向 Point 的指针在堆栈上传递。Interop 封送拆收器不支持任何一个方向上的更高级别的间接寻址 (Point **)

注意

LayoutKind 枚举值设置为 Explicit 的结构不能在 COM Interop 中使用,原因是导出的类型库不能表示显式布局。

系统值类型

System 命名空间具有若干个表示已装箱形式的运行库基元类型的值类型。例如,值类型 System.Int32 结构表示 ELEMENT_TYPE_I4 的已装箱形式。将这些类型以与封送它们所装箱的基元类型相同的方式封送,而不是像其他格式化类型那样作为结构封送。因此,System.Int32 被封送为 ELEMENT_TYPE_I4 而不是包含一个 long 类型的成员的结构。下表包含 System 命名空间中的值类型(它们是基元类型的装箱表示形式)的列表。

系统值类型

元素类型

System.Boolean

ELEMENT_TYPE_BOOLEAN

System.SByte

ELEMENT_TYPE_I1

System.Byte

ELEMENT_TYPE_UI1

System.Char

ELEMENT_TYPE_CHAR

System.Int16

ELEMENT_TYPE_I2

System.UInt16

ELEMENT_TYPE_U2

System.Int32

ELEMENT_TYPE_I4

System.UInt32

ELEMENT_TYPE_U4

System.Int64

ELEMENT_TYPE_I8

System.UInt64

ELEMENT_TYPE_U8

System.Single

ELEMENT_TYPE_R4

System.Double

ELEMENT_TYPE_R8

System.String

ELEMENT_TYPE_STRING

System.IntPtr

ELEMENT_TYPE_I

System.UIntPtr

ELEMENT_TYPE_U

System 命名空间中的其他一些值类型以不同的方式处理。由于非托管代码对这些类型有现成的格式,因此,封送拆收器具有特殊的用于封送它们的规则。下表列出 System 命名空间中的特殊值类型,以及将其封送到的非托管类型。

系统值类型

IDL 类型

System.DateTime

DATE

System.Decimal

DECIMAL

System.Guid

GUID

System.Drawing.Color

OLE_COLOR

下面的代码显示 Stdole2 类型库中的非托管类型 DATEGUIDDECIMAL OLE_COLOR 的定义。

类型库表示形式

typedef double DATE;

typedef DWORD OLE_COLOR;

 

typedef struct tagDEC {

    USHORT    wReserved;

    BYTE      scale;

    BYTE      sign;

    ULONG     Hi32;

    ULONGLONG Lo64;

} DECIMAL;

 

typedef struct tagGUID {

    DWORD Data1;

    WORD  Data2;

    WORD  Data3;

    BYTE  Data4[ 8 ];

} GUID;

下面的代码显示托管 IValueTypes 接口中的相应定义。

public interface IValueTypes {

   void M1(System.DateTime d);

   void M2(System.Guid d);

   void M3(System.Decimal d);

   void M4(System.Drawing.Color d);

}

 

类型库表示形式

interface IValueTypes : IDispatch {

   HRESULT M1([in] DATE d);

   HRESULT M2([in] GUID d);

   HRESULT M3([in] DECIMAL d);

   HRESULT M4([in] OLE_COLOR d);

};

 

转载于:https://www.cnblogs.com/blue-th/archive/2008/09/21/1295180.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值