如何编写 Linq 表达式?即 Func>、Exp>、Exp>>
我正在创建一个 Validator
类。我正在尝试为我的验证器实现 Linq SelectMany
扩展方法,以便能够使用 Linq 查询组成表达式并验证最终结果,即使基础值发生变化也是如此。
下面的测试代码展示了我的意图。
var a = 2;
var b = 3;
var va = Validator.Create(() => a, n => n >= 0 && n < 5);
var vb = Validator.Create(() => b, n => n >= 0 && n < 5);
var vc = from ia in va
from ib in vb
select ia + ib;
Debug.Assert(vc.Value == a + b); //2 + 3
Debug.Assert(vc.Value == 5);
Debug.Assert(vc.IsValid == true);
a = 7;
Debug.Assert(vc.Value == a + b); //7 + 3
Debug.Assert(vc.Value == 10);
Debug.Assert(va.IsValid == false);
Debug.Assert(vb.IsValid == true);
Debug.Assert(vc.IsValid == false);
我看到了以下问题如何编写现有的 Linq 表达式它向我展示了如何使用 And
表达式将两个 Func
组合在一起,但我需要能够以更好的方式将函数组合在一起,功能方式。
例如,我有以下两个表达式:
public Expression<Func<T>> ValueExpression { get; private set; }
public Expression<Func<T, bool>> ValidationExpression { get; private set; }
我希望创建一个如下所示的新表达式:
public Expression<Func<bool>> IsValidExpression
{
get
{
// TODO: Compose expressions rather than compile & invoke.
}
}
更简洁地说,我正在尝试创建这些函数:
// Specific case
Func<Expression<Func<T>>, Expression<Func<T, bool>>, Expression<Func<bool>>>
// General case
Func<Expression<Func<X, Y>>, Expression<Func<Y, Z>>, Expression<Func<X, Z>>>
可以修改通用 case 函数以根据需要接受不同数量的通用参数以进行组合任何功能。
我已经搜索了 Stack Overflow(当然)和网络,但没有解决此问题的示例。
我的 Validator
类的代码如下。
public class Validator<T>
{
public Validator(Expression<Func<T>> valueFunc,
Expression<Func<T, bool>> validationFunc)
{
this.ValueExpression = valueFunc;
this.ValidationExpression = validationFunc;
}
public Expression<Func<T>> ValueExpression { get; private set; }
public Expression<Func<T, bool>> ValidationExpression { get; private set; }
public T Value { get { return this.ValueExpression.Compile().Invoke(); } }
public bool IsValid { get { return this.IsValidExpression.Compile().Invoke(); } }
public Expression<Func<bool>> IsValidExpression
{
get
{
// TODO: Compose expressions.
}
}
}
我的 SelectMany
扩展包含大量令人讨厌的 .Compile().Invoke()
我想摆脱它们。
public static Validator<U> SelectMany<T, U>(this Validator<T> @this, Expression<Func<T, Validator<U>>> k)
{
Expression<Func<T>> fvtv = @this.ValueExpression;
Expression<Func<Validator<U>>> fvu = () => k.Compile().Invoke(fvtv.Compile().Invoke());
Expression<Func<U>> fvuv = fvu.Compile().Invoke().ValueExpression;
Expression<Func<U, bool>> fvtiv = u => @this.ValidationExpression.Compile().Invoke(fvtv.Compile().Invoke());
return fvuv.ToValidator(fvtiv);
}
public static Validator<V> SelectMany<T, U, V>(this Validator<T> @this, Expression<Func<T, Validator<U>>> k, Expression<Func<T, U, V>> s)
{
Expression<Func<Validator<U>>> fvu = () => @this.SelectMany(k);
Expression<Func<T>> fvtv = @this.ValueExpression;
Expression<Func<U>> fvuv = fvu.Compile().Invoke().ValueExpression;
Expression<Func<T, bool>> fvtiv = @this.ValidationExpression;
Expression<Func<U, bool>> fvuiv = u => fvu.Compile().Invoke().ValidationExpression.Compile().Invoke(u);
Expression<Func<V>> fvv = () => s.Compile().Invoke(fvtv.Compile().Invoke(), fvuv.Compile().Invoke());
Expression<Func<V, bool>> fvviv = v => fvtiv.Compile().Invoke(fvtv.Compile().Invoke()) && fvuiv.Compile().Invoke(fvuv.Compile().Invoke());
return fvv.ToValidator(fvviv);
}
提前致谢!
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
虽然dtb的答案适用于多种场景,但它不是最佳的,因为这样的表达式不能在实体框架中使用,因为它无法处理
Invoke
调用。不幸的是,为了避免这些调用,需要更多的代码,包括一个新的ExpressionVisitor
派生类:这会将第一个表达式中第一个参数的每个实例替换为第二个表达式中的表达式。因此,像这样的调用:
Compose((Class1 c) => c.StringProperty, (Class2 c2) => c2.Class1Property
将产生表达式
(Class2 c2) =>; c2.Class1Property.StringProperty
。While dtb's answer works for several scenarios, it is suboptimal as such an expression cannot be used in Entity Framework, as it cannot handle
Invoke
calls. Unfortunately, to avoid those calls one needs a lot more code, including a newExpressionVisitor
derived class:This replaces every instance of the first parameter in the first expression with the expression in the second expression. So a call like this:
Compose((Class1 c) => c.StringProperty, (Class2 c2) => c2.Class1Property
Would yield the expression
(Class2 c2) => c2.Class1Property.StringProperty
.在 C# 中,Haskell 的函数组合运算符的等效项
可能类似于“
这是您正在寻找的吗?”
例子:
The equivalent of Haskell's function composition operator
would in C# probably be something like
Is this what you're looking for?
Example: