C#命令获取struct offset?

Pau*_*rth 8 c#

假设我有一个像这样的C#结构:

[StructLayout(LayoutKind.Explicit)]
struct IMAGE_DOS_HEADER {
    [FieldOffset(60)] public int e_lfanew;
}
Run Code Online (Sandbox Code Playgroud)

现在假设我从文件中读取数据,如下所示:

byte[] data = new byte[4096];
FileStream f = new FileInfo(filename).Open(FileMode.Open, FileAccess.Read);
int n = f.Read(data, 0, 4096);
Run Code Online (Sandbox Code Playgroud)

现在我想测试n以确保我已经读取了足够的字节来获取值e_lfanew.有没有什么办法可以获得值60(FieldOffset)而无需重新输入?我正在寻找这样的东西:

if (n >= offsetof(IMAGE_DOS_HEADER.e_lfanew) + sizeof(int)) {
    ...
}
Run Code Online (Sandbox Code Playgroud)

有这样的命令吗?在我的实际代码中,我必须进行其中几个测试,并且通过在结构中添加先前字段或通过从FieldOffset属性中复制值来手动输入数字似乎很乏味且容易出错.有没有更好的办法?

wj3*_*j32 18

使用Marshal.OffsetOf:

Marshal.OffsetOf(typeof(IMAGE_DOS_HEADER), "e_lfanew")
Run Code Online (Sandbox Code Playgroud)

  • 为什么要知道托管结构偏移?即使你知道你无法以任何方式使用它,因为即使获得指向结构的指针也意味着它是快速的.您的观察与任何实际目的完全无关. (5认同)
  • 我不知道这个;“Marshal”用于在托管和非托管之间切换。因此,从文档中“OffsetOf”提供了非托管结构布局的偏移量,**这不一定对应于托管结构布局的偏移量**。因此在某些情况下这可能会给出错误的答案。 (3认同)
  • Marshal.OffsetIf 可以用于结构体中的字段/属性吗?这需要如何命名?让我们说:Struct1.Struct2.Prop1 Strct1 中 Prop1 的标识符是什么? (2认同)

jas*_*son 7

是的,你可以用反射做到这一点.

FieldOffsetAttribute fieldOffset = 
    (FieldOffsetAttribute)typeof(IMAGE_DOS_HEADER)
        .GetField("e_lfanew")
        .GetCustomAttributes(
            typeof(FieldOffsetAttribute),
            true
        )[0];
Console.WriteLine(fieldOffset.Value);
Run Code Online (Sandbox Code Playgroud)

你甚至可以把它变成一个很好的方法:

static int FieldOffset<T>(string fieldName) {
    if (typeof(T).IsValueType == false) {
        throw new ArgumentOutOfRangeException("T");
    }
    FieldInfo field = typeof(T).GetField(fieldName);
    if (field == null) {
        throw new ArgumentOutOfRangeException("fieldName");
    }
    object[] attributes = field.GetCustomAttributes(
        typeof(FieldOffsetAttribute),
        true
    );
    if (attributes.Length == 0) {
        throw new ArgumentException();
    }
    FieldOffsetAttribute fieldOffset = (FieldOffsetAttribute)attributes[0];
    return fieldOffset.Value;
}
Run Code Online (Sandbox Code Playgroud)

用法:

Console.WriteLine(FieldOffset<IMAGE_DOS_HEADER>("e_lfanew"));
Run Code Online (Sandbox Code Playgroud)


Han*_*ant 7

好吧,你已经在结构声明中使用了一个幻数.不妨这样做:

private const int LfaNewOffset = 60;

[StructLayout(LayoutKind.Explicit)]
struct IMAGE_DOS_HEADER {
    [FieldOffset(LfaNewOffset)] public int e_lfanew;
}
...
if (n >= LfaNewOffset + sizeof(int)) {
    ...
}
Run Code Online (Sandbox Code Playgroud)