如何使用反射设置固定缓冲区域元素?

时间:2015-06-04 13:34:05

标签: c# reflection unsafe

这是一个典型的unsafe结构声明,其中包含一个固定的缓冲区字段:

[StructLayout(LayoutKind.Explicit, Pack = 1)]
public unsafe struct MyStruct
{
    ...

    [FieldOffset(6)]
    public fixed ushort MyFixedBuffer[28];

    ...
}

如何使用反射设置MyFixeBuffer的元素?

1 个答案:

答案 0 :(得分:3)

你不需要反思。您需要的是要设置的结构的地址。然后你可以使用

byte* pStruct = xxxx
byte* pMyFixedBuffer = pStruct+6;  // FieldOffset 6 tells me this

*(pMyFixedBuffer+i) = yourBytevalue;  // set i-th element in MyFixedBuffer

我猜你在获取结构地址方面遇到了问题,因为它很可能通过值传递给你想要修补不同值的某些方法。只要结构未分配给类成员变量,您可以从外部以某种方式访问​​该实例,您将无法在运行时修补任何内容。

由于您的课程是公开的,并且该字段也是公开的,因此没有理由使用反射或不安全的代码。但我想这个课程并不是你正在努力的实际课程。

您可以使用反射来查找类布局,但在某些时候您需要硬编码您实际想要修补的字段。为此,您可以使用从反射中获取的布局信息,然后在运行时确定地址。您需要获取该字段的FieldOffset属性值,然后将其用作指针偏移量来获取它。

请注意,数组MyFixedBuffer不是结构中的真实数组,因为它嵌入在结构中。通常的对象指针GetType的东西不起作用,因为数组不受管理,而是一个固定大小的缓冲区,根本没有MT(方法表指针)。此时,您需要处理您所追求的字节中的原始偏移量和补丁。

下面是一个使用反射的示例,如果您想知道如何处理任何值类型,如果您想构建使用值类型的动态内容,或者例如一个序列化器。

[StructLayout(LayoutKind.Explicit, Pack = 1)]
public unsafe struct MyStruct
{

    [FieldOffset(6)]
    public fixed ushort MyFixedBuffer[28];

    [FieldOffset(62)]
    public byte Next;
}

class Program
{
    unsafe static void Main(string[] args)
    {
        var field = typeof(MyStruct).GetField("MyFixedBuffer");
        int offset = field.CustomAttributes.Where(x => x.AttributeType.Equals(typeof(FieldOffsetAttribute)))
                                                .SelectMany(x => x.ConstructorArguments)
                                                .Select(x => x.Value)
                                                .Cast<int>()
                                                .First();

        KeyValuePair<Type,int> fixedBufferDescription = field.CustomAttributes.Where(x => x.AttributeType.Equals(typeof(FixedBufferAttribute)))
                                                   .Select(x => x.ConstructorArguments)
                                                   .Select(x => new KeyValuePair<Type, int>((Type)x[0].Value, (int)x[1].Value))
                                                   .First();

        int fixedBuferLen = Marshal.SizeOf(fixedBufferDescription.Key) * fixedBufferDescription.Value;


        MyStruct tmp = new MyStruct();
        byte* raw = (byte*) &tmp;
        short* pmyArray = (short *) (raw + offset);

        for (int i = 0; i < fixedBuferLen/Marshal.SizeOf(fixedBufferDescription.Key); i++)
        {
            *(pmyArray + i) = (short) i;
        }

    }
}

这应该可以解决问题。