DEVMODE结构包含两个联合,要使用精确的字节偏移量,才能将每个结构成员正确排序。
这里使用Ansi和unicode两个版本,对应以A或W结尾的EnumDisplaySettings函数,如果不对应各自版本的函数将出现不正确的结构,在Ansi,一个字符为1,在unicode,则为2。DeviceName在这里版本的字节偏移量是不同的,Ansi版本占了32*1,即32个偏移量;unicode版本占了32*2,即64个偏移量。另,这里使用了Pack=2,按2对齐,因此DeviceName还要填充2个字节,所以占了34或者66。
关于C#中使用联合,只要在结构中的几个字段成员具有相同的偏移量,那么它们就共享一个内存地址,该地址的偏移量取在这个共享的地址中,字节长度最大的那个成员的长度,另外,必须使用:LayoutKind.Explicit,用户自己定义每个字段成员的偏移量。
例:
[StructLayout(LayoutKind.Explicit, Pack = 2, CharSet = CharSet.Ansi)]
public struct Mystruct{
[FieldOffset(0)]
public char c;
[FieldOffset(0)]
public POINTL Union1_POINTL;//POINTL 为一个结构包含两个int类型
[FieldOffset(0)]
public int Union1_DisplayOrientation;
[FieldOffset(0)]
public int Union1_DisplayFixedOutput;
};
如果Marshal.SizeOf(typeof(Mystruct));等于POINTL的长度,为8
好了,闲话不多说,下面声明DEVMODE结构的两个版本:
由于DEVMODE结构中包含一个匿名联合,这个匿名联合又包含一个匿名结构,c#不能使用匿名结构,因此在这里将该匿名结构的成员需要声明一个结构,将这些成员包起来。这里声明为Anonymity:
/// <summary>
/// 此结构是API的DEVMODE结构在联合里的一个匿名结构
/// </summary>
[StructLayout(LayoutKind.Explicit, Pack = 2)]
public struct Anonymity {
[FieldOffset(0)]
public short dmOrientation;
[FieldOffset(2)]
public short dmPaperSize;
[FieldOffset(4)]
public short dmPaperLe