2

我是 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,编译它,然后缓存它。提前致谢。

4

1 回答 1

4

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

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

这几乎与您所说的一样,除了它碰巧使用ILGeneratorvia 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; }
}
于 2012-09-28T05:15:56.503 回答