【发布时间】:2015-01-26 08:09:14
【问题描述】:
我已经看到了 Issue while building dynamic Expression Tree 和 Expression/Statement trees,由于我是表达式树的新手,我仍在努力理解如何实现我想要的。
一个人为的对象在下面
public class TestObject
{
public TestObject()
{
ClassList = new List<Class>();
}
public int Age { get; set; }
public List<Class> ClassList { get; set; }
}
public class Class
{
public string Name { get; set; }
public int ClassId { get; set; }
}
在运行时,我遍历每个属性并生成一个委托,该委托将转换为该属性的字符串。我已经完成了所有工作。我现在必须处理的问题是,对于 List 类型,我需要能够对 ClassList 属性中的每个项目应用一组操作,因此我需要一个允许我这样做的 foreach。
我现在有这个
//type==TestObject at runtime
//propertyName == "ClassList"
ParameterExpression recordExpression = Expression.Parameter(type, "record");
memberExpression = MemberExpression.Property(recordExpression, propertyName);
Type getEnumerableDelegateType =
typeof(Func<,>).MakeGenericType(new Type[] { type, memberExpression.Type});
var getList = Expression.Lambda(getEnumerableDelegateType, memberExpression, recordExpression);
GetList 在编译和调用时会按预期返回 List。我正在努力的是如何创建一个表达式,该表达式将使用 lambda 表达式的结果并应用我已经为每个 Class 项创建的一组操作对其进行迭代。
最终我正在寻找一个 lambda 签名来匹配下面的 overallAction 签名
var getListFunc = new Func<TestObject, List<Class>>((TestObject obj1) => obj1.ClassList);
Action<List<Class>> listAction = delegate(List<Class> data)
{
foreach (var dataChannelWithUnitse in data)
{
//Apply generated delegate
}
};
Action<TestObject> overallAction = delegate(TestObject data)
{
var x = getListFunc.Invoke(data);
listAction.Invoke(x as List<Class>);
};
感谢任何帮助以帮助我了解如何执行此操作。
我目前有这个例外,从范围''引用的'TestObject'类型的变量'Input',但它没有定义
var typeParam = Expression.Parameter(type, "Input");
var listVariable = Expression.Variable(memberExpression.Type, "List");
var enumerator = Expression.Variable(typeof(IEnumerator<>).MakeGenericType(dataType));
var enumeratorType = typeof(IEnumerator<>).MakeGenericType(dataType);
var enumerableType = typeof(IEnumerable<>).MakeGenericType(dataType);
var enumerableParam = Expression.Parameter(enumerableType, "ExtractedCollection");
var getEnumeratorFunc = Expression.Call(enumerableParam, enumerableType.GetMethod("GetEnumerator"));
var getEnumeratorLambda = Expression.Lambda(getEnumeratorFunc, enumerableParam);
var t1 = Expression.Assign(listVariable, Expression.Invoke(getListLambda, typeParam));
var t2 = Expression.Assign(enumerator, Expression.Invoke(getEnumeratorLambda, listVariable));
var @break = Expression.Label();
var funcBlock = Expression.Block(
new ParameterExpression[] { listVariable, enumerator},
t1,
t2,
Expression.Loop(
Expression.IfThenElse(
Expression.NotEqual(Expression.Call(enumerator,typeof(IEnumerator).GetMethod("MoveNext")),Expression.Constant(false)),
Expression.Invoke(enumerableExpressions[0],Expression.Property(enumerator, "Current")),
Expression.Break(@break))
, @break), typeParam);
Expression<Action<TestObject>> lm = Expression.Lambda<Action<TestObject>>(funcBlock,recordExpression);
var d = lm.Compile(); **//this is exceptioning with " variable 'Input' of type 'TestObject' referenced from scope '', but it is not defined**
【问题讨论】:
-
List<T>实际上有一个您可以调用的.ForEach()方法。这并不适用于任何IEnumerable,但在这种特殊情况下,它可以帮助您简化代码。
标签: c# .net lambda expression-trees