3

明示的な構造体レイアウトと構造体オーバーレイを理解しようとしていますが、期待する動作が見られません。以下のコードを考えると:

class Program
{

    static void Main(string[] args)
    {
        byte[] bytes = new byte[17];
        bytes[0] = 0x01; // Age is 1    //IntField1
        bytes[1] = 0x00;                //IntField1
        bytes[2] = 0x00;                //IntField1
        bytes[3] = 0x00;                //IntField1
        bytes[4] = 0x02;                //IntField2
        bytes[5] = 0x00;                //IntField2
        bytes[6] = 0x00;                //IntField2
        bytes[7] = 0x00;                //IntField2

        bytes[8] = 0x41;                //CharArray A
        bytes[9] = 0x42;                //CharArray B
        bytes[10] = 0x43;               //CharArray C
        bytes[11] = 0x44;               //CharArray D

        bytes[12] = 0x45;               //CharArray E

        bytes[13] = 0x46;               //CharArray F
        bytes[14] = 0x00; // \0 decimal 0
        bytes[15] = 0x00; // \0 decimal 0
        bytes[16] = 0x01; // 1 decimal 1
        Console.WriteLine(Marshal.SizeOf(typeof(TestStruct)));

        TestStruct testStruct2 = (TestStruct) RawDeserialize(bytes, 0, typeof (TestStruct));

        Console.WriteLine(testStruct2);
        Console.ReadLine();
    }
    public static object RawDeserialize( byte[] rawData, int position, Type anyType )
    {
        int rawsize = Marshal.SizeOf( anyType );
        if( rawsize > rawData.Length )
            return null;

        IntPtr buffer = Marshal.AllocHGlobal( rawsize );
        Marshal.Copy( rawData, position, buffer, rawsize );
        object retobj = Marshal.PtrToStructure( buffer, anyType );
        Marshal.FreeHGlobal( buffer );
        return retobj;
    }
}

[StructLayout(LayoutKind.Explicit, Pack = 1)]
public struct TestStruct
{
    [FieldOffset(0)]
    public int IntField1;
    [FieldOffset(4)]
    public int IntField2;
    [FieldOffset(8)]
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]
    public char[] CharArray;
    [FieldOffset(16)]
    public byte SomeByte;        

    [FieldOffset(8)]
    public TestStruct2 SubStruct;

    public override string ToString()
    {
        return string.Format("IntField1: {0}\nIntField2: {1}\nCharArray: {2}\nSomeByte: {3}\nSubStruct:\n{{{4}}}", 
            IntField1, IntField2,  new string(CharArray), SomeByte, SubStruct);
    }
}

[StructLayout(LayoutKind.Explicit, Pack = 1)]
public struct TestStruct2
{
    [FieldOffset(0)]
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]
    public char[] CharArray1;
    [FieldOffset(0)]
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)]
    public char[] CharArray2;
    [FieldOffset(4)]
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 2)]
    public char[] CharArray3;

    public override string ToString()
    {
        return string.Format("CharArray1: {0}\nCharArray2: {1}\nCharArray3: {2}",
           new string(CharArray1), new string(CharArray2), new string(CharArray3));
    }
}

これによる結果は次のようになると思います。

IntField1:1
IntField2:2
CharArray:ABCDEF
SomeByte:1
SubStruct:
{CharArray1:ABCDEF
CharArray2:ABCD
CharArray3:E}

しかし、結果は次のとおりです。

IntField1:1
IntField2:2
CharArray:ABCD
SomeByte:1
SubStruct:
{CharArray1:ABCD
CharArray2:ABCD
CharArray3:EF}

TestStructのCharArrayの長さが4であるのはなぜですか?ABCDEFが6文字であると想定しましたが、ABCDしか含まれていません。TestStruct2.CharArray1についても同じです。

4

3 に答える 3

3

TestStruct2をCharArrayの後に、同じオフセットに配置することにより、TestStruct2のCharArray2へのポインターは、TestStruct自体のchararrayへのポインターであったものを上書きします。

TestStruct2のCharArray2の長さをコメントアウトまたは変更すると、期待される結果が表示されます。

同様に、struct2を最初に配置したときに何が起こるかを見てください。

[StructLayout(LayoutKind.Explicit, Pack = 1)]
    public struct TestStruct
    {
        [FieldOffset(8)]
        public TestStruct2 SubStruct;

        [FieldOffset(0)]
        public int IntField1;
        [FieldOffset(4)]
        public int IntField2;
        [FieldOffset(8)]
        [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]
        public char[] CharArray;
        [FieldOffset(16)]
        public byte SomeByte;        


        public override string ToString()
        {
            return string.Format("IntField1: {0}\nIntField2: {1}\nCharArray: {2}\nSomeByte: {3}\nSubStruct:\n{{{4}}}", 
                IntField1, IntField2,  new string(CharArray), SomeByte, SubStruct);
        }
    }

これで効果が逆になり、TestStruct2のCharArray2の長さは6文字になります。

于 2011-02-25T03:40:58.323 に答える
0

覚えておくべきことの 1 つは、C# の char は 2 バイトの Unicode 文字であることです。

サブ構造体をそこに置いたまま、期待した結果を得ることができませんでしたが。オーバーラップしている SizeConst 配列が混乱しているようです。

于 2011-03-07T06:10:49.157 に答える