访问 IEnumerable孩子们
这是我们想要做的。
我们有来自数据库的数据,需要对其进行格式化以制作报告,包括一些计算(总和、平均值和字段到字段计算(例如:xa / xb))。
限制之一是,例如,在求和中,如果其中一个数据为空、-1 或 -2,我们必须停止计算并显示“-”。由于我们要生成许多报告,每个报告都具有相同的逻辑和许多计算,因此我们希望集中此逻辑。目前,我们生成的代码允许我们检查字段到字段的计算(例如 xa / xb),但不允许我们检查组总计(例如:xb / SUM(xa))
测试用例
规则
- 计算如果计算中使用的值之一是 -1、-2 或 null,则不应执行。在这种情况下,如果找到 -1 或 null,则返回“-”;如果找到 -2,则返回“C”。
- 如果计算中有多个“坏值”,则需要遵循如下定义的优先级: null -> ; -1-> -2。此优先级与计算中值所在的级别无关
测试
简单计算
object: new DataInfo { A = 10, B = 2, C = 4 } calcul: x => x.A / x.B + x.C result: 9
object: new DataInfo { A = 10, B = 2, C = -2 } calcul: x => x.A / x.B + x.C result: C (because you have a '-2' value in the calcul)
object: new DataInfo { A = 10, B = -2, C = null } calcul: x => x.A / x.B + x.C result: - (because you have a 'null' value in the calcul and it win on the -2 value)
复杂计算
object: var list = new List(); list.Add(new DataInfo { A = 10, B = 2, C = 4 }); list.Add(new DataInfo { A = 6, B = 3, C = 2 }); calcul: list.Sum(x => x.A / x.B + list.Max(y => y.C)) result: 15
object: var list = new List(); list.Add(new DataInfo { A = 10, B = 2, C = 4 }); list.Add(new DataInfo { A = 6, B = 3, C = -2 }); calcul: list.Sum(x => x.A / x.B + list.Max(y => y.C)) result: C (because you have a '-2' value in the calcul)
到目前为止我们做了什么
这是我们必须基于此线程处理简单计算的代码:
如何提取属性在表达式
我们创建了一个强类型类,它执行计算并将结果作为字符串返回。但如果表达式的任何部分等于特殊值,则计算器必须返回特殊字符。
它适用于简单的情况,就像这样:
var data = new Rapport1Data() { UnitesDisponibles = 5, ... };
var q = new Calculator<Rapport1Data>()
.Calcul(data, y => y.UnitesDisponibles, "N0");
但我需要能够执行一些更复杂的事情,例如:
IEnumerable<Rapport1Data> data = ...;
var q = new Calculator<IEnumerable<Rapport1Data>>()
.Calcul(data, x => x.Sum(y => y.UnitesDisponibles), "N0");
当我们开始在 IEnurmarable<>
中封装数据时,我们会收到一个错误:
对象与目标类型不匹配
正如我们所理解的,这是因为子表达式 y =>; y.UnitesDisponibles
应用于 IEnumerable
而不是 Rapport1Data
。
如果有一天我们有复杂的表达式,例如
IEnumerable<IEnumerable<Rapport1Data>> data = ...;
var q = new Calculator<IEnumerable<IEnumerable<Rapport1Data>>>()
.Calcul(data,x => x.Sum(y => y.Sum(z => z.UnitesDisponibles)), "N0");
我们构建的类
public class Calculator<T>
{
public string Calcul(
T data,
Expression<Func<T, decimal?>> query,
string format)
{
var rulesCheckerResult = RulesChecker<T>.Check(data, query);
// l'ordre des vérifications est importante car il y a une gestion
// des priorités des codes à retourner!
if (rulesCheckerResult.HasManquante)
{
return TypeDonnee.Manquante.ReportValue;
}
if (rulesCheckerResult.HasDivisionParZero)
{
return TypeDonnee.DivisionParZero.ReportValue;
}
if (rulesCheckerResult.HasNonDiffusable)
{
return TypeDonnee.NonDiffusable.ReportValue;
}
if (rulesCheckerResult.HasConfidentielle)
{
return TypeDonnee.Confidentielle.ReportValue;
}
// if the query respect the rules, apply the query and return the
// value
var result = query.Compile().Invoke(data);
return result != null
? result.Value.ToString(format)
: TypeDonnee.Manquante.ReportValue;
}
}
和自定义 ExpressionVisitor
class RulesChecker<T> : ExpressionVisitor
{
private readonly T data;
private bool hasConfidentielle = false;
private bool hasNonDiffusable = false;
private bool hasDivisionParZero = false;
private bool hasManquante = false;
public RulesChecker(T data)
{
this.data = data;
}
public static RulesCheckerResult Check(T data, Expression expression)
{
var visitor = new RulesChecker<T>(data);
visitor.Visit(expression);
return new RulesCheckerResult(
visitor.hasConfidentielle,
visitor.hasNonDiffusable,
visitor.hasDivisionParZero,
visitor.hasManquante);
}
protected override Expression VisitBinary(BinaryExpression node)
{
if (!this.hasDivisionParZero &&
node.NodeType == ExpressionType.Divide &&
node.Right.NodeType == ExpressionType.MemberAccess)
{
var rightMemeberExpression = (MemberExpression)node.Right;
var propertyInfo = (PropertyInfo)rightMemeberExpression.Member;
var value = Convert.ToInt32(propertyInfo.GetValue(this.data, null));
this.hasDivisionParZero = value == 0;
}
return base.VisitBinary(node);
}
protected override Expression VisitMember(MemberExpression node)
{
// Si l'un d'eux n'est pas à true, alors continuer de faire les tests
if (!this.hasConfidentielle ||
!this.hasNonDiffusable ||
!this.hasManquante)
{
var propertyInfo = (PropertyInfo)node.Member;
object value = propertyInfo.GetValue(this.data, null);
int? valueNumber = MTO.Framework.Common.Convert.To<int?>(value);
// Si la valeur est à true, il n'y a pas lieu de tester davantage
if (!this.hasManquante)
{
this.hasManquante =
valueNumber == TypeDonnee.Manquante.BdValue;
}
// Si la valeur est à true, il n'y a pas lieu de tester davantage
if (!this.hasConfidentielle)
{
this.hasConfidentielle =
valueNumber == TypeDonnee.Confidentielle.BdValue;
}
// Si la valeur est à true, il n'y a pas lieu de tester davantage
if (!this.hasNonDiffusable)
{
this.hasNonDiffusable =
valueNumber == TypeDonnee.NonDiffusable.BdValue;
}
}
return base.VisitMember(node);
}
}
[更新], 我们如何修复它以确保它是完全递归的 添加更多关于我们想要做的事情的细节
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
您需要更改一些内容才能使其正常工作:
这是我认为需要完成的工作的一个粗略示例。
There are a few things that you need to change to get this to work:
Here is a rough example of what I think needs to be done.
如果我理解正确的话,您正在寻找“递归求和”函数。我可以建议这样的事情吗?
*注意:代码未经测试,甚至可能无法编译
If I understand correctly you're looking for a 'recursive sum' function. Might I suggest something like this?
*note: code not tested, might not even compile