这是一个典型的unsafe
结构声明,其中包含一个固定的缓冲区字段:
[StructLayout(LayoutKind.Explicit, Pack = 1)]
public unsafe struct MyStruct
{
...
[FieldOffset(6)]
public fixed ushort MyFixedBuffer[28];
...
}
如何使用反射设置MyFixeBuffer
的元素?
答案 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;
}
}
}
这应该可以解决问题。