将反射方法转换为已编译的lambda

时间:2012-09-28 05:07:16

标签: c# reflection lambda

我是lambda的新手。如果我的问题很简单,请原谅。

我有一个方法,它使用反射在某些类型上设置属性:

public void WriteId(object obj, int id) {
    var type = obj.GetType();
    var prop = type.GetProperties(BindingFlags.Public | BindingFlags.Instance)
                   .Where(p => p.CanRead && p.CanWrite)
                   .Where(p => p.Name == "Id")
                   .Where(p.PropertyType == typeof(int))
                   .FirstOrDefault();
    if(prop != null)
        prop.SetValue(obj, id, null);
}

你能告诉我,我怎样才能创建一个同样工作的lambda?实际上我想为每种类型创建一个lambda,编译它并缓存它。提前谢谢。

1 个答案:

答案 0 :(得分:4)

我会从NuGet安装FastMember,然后使用:

var wrapped = ObjectAccessor.Create(obj);
obj["Id"] = id;

除了恰好通过ILGenerator(而不是TypeBuilder)使用Expression之外,它几乎与您说的相同 - 但所有缓存等都在那里。

第二种不正当的方法是让dynamic为你完成所有这些:

((dynamic)obj).Id = id;

但是如果你想出于其他原因使用Expression

using System;
using System.Linq.Expressions;   

static class Program
{
    static void Main()
    {
        var obj = new Foo { Id = 2 };
        WriteId(obj, 6);
        Console.WriteLine(obj.Id); // 6
    }

    private static class SneakyCache<T>
    {
        public static readonly Action<T, int> SetId;
        static SneakyCache()
        {
            var obj = Expression.Parameter(typeof(T), "obj");
            var id = Expression.Parameter(typeof(int), "id");
            SetId = Expression.Lambda<Action<T, int>>(
                  Expression.Assign(Expression.Property(obj, "Id"), id),
                  obj, id).Compile();
        }
    }
    public static void WriteId<T>(T obj, int id) where T : class
    {
        SneakyCache<T>.SetId(obj, id);
    }
}
class Foo
{
    public int Id { get; set; }
}