我是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,编译它并缓存它。提前谢谢。
答案 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; }
}