使用反射(C#)获取字段的内存地址

使用反射(C#)获取字段的内存地址,c#,.net-core,C#,.net Core,在.NETCore中,当父对象在编译时未知时,如何获取基元字段的内存地址(IntPtr) 如果对象类是struct和blittable,那么我可以使用Marshel.Offset,但不幸的是它不是 下面的代码说明了我正在尝试做什么 class Example // this class is defined in a different assembly and is not known at compile time { public double foo;

在.NETCore中,当父对象在编译时未知时,如何获取基元字段的内存地址(IntPtr)

如果对象类是struct和blittable,那么我可以使用Marshel.Offset,但不幸的是它不是

下面的代码说明了我正在尝试做什么

    class Example   // this class is defined in a different assembly and is not known at compile time
    {
        public double foo;
    }

    static void Main(string[] args)
    {
        Example obj = new Example();
        obj.foo = 123;

        var hdl = GCHandle.Alloc(obj, GCHandleType.Pinned);
        IntPtr foo_add = GetAddressOf(obj, "foo");

        hdl.Free();
    }

    static IntPtr GetAddressOf(object pinned_object, string field_name)
    {
        FieldInfo field = pinned_object.GetType().GetField(field_name);
        return field.GetFieldAddress(pinned_object); // Unfortunatly this function does not exist, what are the alternatives?
    }

另一种方法是获取对象的内存地址,这可以使用GCHandle.Alloc完成,但不适用于原语。

下面的答案描述了如何使用C#7.0中的“ref return”功能获取对字段的托管引用:

基于这个出色的答案,我们可以修改函数
create\u refgetter
,以获得非托管指针。这利用了鲜为人知的函数uu makeref,该函数在处理泛型类型()时是必需的

然后使用函数
GetAddressOf
包装此函数,该函数允许在不指定泛型类型的情况下调用

    static IntPtr GetAddressOf(object pinned_object, string field_name)
    {
        var fi_val = pinned_object.GetType().GetField(field_name);
        var mi_all = typeof(Program).GetMethod("create_refgetter", BindingFlags.Static | BindingFlags.Public);

        var mi_generic = mi_all.MakeGenericMethod(pinned_object.GetType(), fi_val.FieldType);
        var ptr = (IntPtr) mi_generic.Invoke(null, new object[] {pinned_object, field_name });

        return ptr;
    }

    https://stackoverflow.com/a/45046664/425678

    public delegate ref U RefGetter<T, U>(T obj);
    public static IntPtr create_refgetter<T, U>(object obj,string s_field)
    {
        const BindingFlags bf = BindingFlags.NonPublic |
                                BindingFlags.Public |
                                BindingFlags.Instance |
                                BindingFlags.DeclaredOnly;

        var fi = typeof(T).GetField(s_field, bf);
        if (fi == null)
            throw new MissingFieldException(typeof(T).Name, s_field);

        var s_name = "__refget_" + typeof(T).Name + "_fi_" + fi.Name;

        // workaround for using ref-return with DynamicMethod:
        //   a.) initialize with dummy return value
        var dm = new DynamicMethod(s_name, typeof(U), new[] { typeof(T) }, typeof(T), true);

        //   b.) replace with desired 'ByRef' return value
        dm.GetType().GetField("m_returnType", bf).SetValue(dm, typeof(U).MakeByRefType());

        var il = dm.GetILGenerator();
        il.Emit(OpCodes.Ldarg_0);
        il.Emit(OpCodes.Ldflda, fi);
        il.Emit(OpCodes.Ret);

        RefGetter<T, U> ref_getter = (RefGetter<T, U>)dm.CreateDelegate(typeof(RefGetter<T, U>));

        unsafe
        {
            TypedReference t_ref = __makeref(ref_getter((T)obj));

            IntPtr ptr = *((IntPtr*)&t_ref);
            return ptr;
        }
    }
}
示例输出:

Current Value: 123
Updated Value: 4
很明显,它甚至不接近复制品。Op询问了一个基本字段的地址。另一个问题是对象的地址。
Current Value: 123
Updated Value: 4