Esta es una versión simplificada del problema original.
Tengo una clase llamada Persona:
public class Person {
public string Name { get; set; }
public int Age { get; set; }
public int Weight { get; set; }
public DateTime FavouriteDay { get; set; }
}
... y digamos una instancia:
var bob = new Person {
Name = "Bob",
Age = 30,
Weight = 213,
FavouriteDay = '1/1/2000'
}
Me gustaría escribir lo siguiente como una cadena en mi editor de texto favorito ...
(Person.Age > 3 AND Person.Weight > 50) OR Person.Age < 3
Me gustaría tomar esta cadena y mi instancia de objeto y evaluar un VERDADERO o FALSO, es decir, evaluar un Func <Person, bool> en la instancia del objeto.
Aquí están mis pensamientos actuales:
- Implemente una gramática básica en ANTLR para admitir operadores lógicos y de comparación básicos. Estoy pensando en copiar la precedencia de Visual Basic y algunas de las características aquí: http://msdn.microsoft.com/en-us/library/fw84t893(VS.80).aspx
- Haga que ANTLR cree un AST adecuado a partir de una cadena proporcionada.
- Recorre el AST y usa el Generador de predicados marco para crear dinámicamente el Func <Person, bool>
- Evaluar el predicado contra una instancia de Persona según sea necesario
Mi pregunta es ¿he superado totalmente esto? alguna alternativa?
EDITAR: Solución elegida
Decidí usar la Biblioteca Dynamic Linq, específicamente la clase Dynamic Query provista en LINQSamples.
Código a continuación:
using System;
using System.Linq.Expressions;
using System.Linq.Dynamic;
namespace ExpressionParser
{
class Program
{
public class Person
{
public string Name { get; set; }
public int Age { get; set; }
public int Weight { get; set; }
public DateTime FavouriteDay { get; set; }
}
static void Main()
{
const string exp = @"(Person.Age > 3 AND Person.Weight > 50) OR Person.Age < 3";
var p = Expression.Parameter(typeof(Person), "Person");
var e = System.Linq.Dynamic.DynamicExpression.ParseLambda(new[] { p }, null, exp);
var bob = new Person
{
Name = "Bob",
Age = 30,
Weight = 213,
FavouriteDay = new DateTime(2000,1,1)
};
var result = e.Compile().DynamicInvoke(bob);
Console.WriteLine(result);
Console.ReadKey();
}
}
}
El resultado es del tipo System.Boolean, y en este caso es VERDADERO.
Muchas gracias a Marc Gravell.
Incluya el paquete System.Linq.Dynamics nuget, documentación aquí
Respuestas:
¿ La biblioteca dinámica de linq ayudaría aquí? En particular, estoy pensando como una
Where
cláusula. Si es necesario, ¡colóquelo dentro de una lista / matriz solo para llamarlo.Where(string)
! es decirSi no, escribir un analizador sintáctico (usando
Expression
debajo del capó) no es muy exigente: escribí uno similar (aunque no creo que tenga la fuente) en mi viaje en tren justo antes de Navidad ...fuente
// Lambda expression as data in the form of an expression tree.
System.Linq.Expressions.Expression<Func<int, bool>> expr = i => i < 5;
// Compile the expression tree into executable code.
Func<int, bool> deleg = expr.Compile();
// Invoke the method and print the output.
Console.WriteLine("deleg(4) = {0}", deleg(4));
ParseLambda bueno!Otra biblioteca de este tipo es Flee
Hice una comparación rápida de Dynamic Linq Library y Flee and Flee fue 10 veces más rápido para la expresión
"(Name == \"Johan\" AND Salary > 500) OR (Name != \"Johan\" AND Salary > 300)"
Así es como puedes escribir tu código usando Flee.
fuente
LinqPad tiene el
Dump()
métodofuente
var type = typeof(T); var prop = type.GetProperty(propName);
que se compilara.Puede echar un vistazo a la DLR . Le permite evaluar y ejecutar scripts dentro de la aplicación .NET 2.0. Aquí hay una muestra con IronRuby :
Por supuesto, esta técnica se basa en la evaluación del tiempo de ejecución y el código no se puede verificar en tiempo de compilación.
fuente
Aquí hay un ejemplo de un combinador de analizador basado en DSL Scala para analizar y evaluar expresiones aritméticas.
El árbol de expresión equivalente o el árbol de análisis de la expresión aritmética proporcionada sería del tipo Parser [List [String]].
Más detalles en el siguiente enlace:
http://nicolaecaralicea.blogspot.ca/2013/04/scala-dsl-for-parsing-and-evaluating-of.html
fuente
Además de Dynamic Linq Library (que construye expresiones fuertemente tipadas y requiere variables fuertemente tipadas), recomiendo una mejor alternativa: linq parser esa parte de NReco Commons Library (código abierto). Alinea todos los tipos y realiza todas las invocaciones en tiempo de ejecución y se comporta como un lenguaje dinámico:
fuente
Aunque esta es una publicación relativamente antigua, este es el código para el generador de expresiones: AnyService - ExpressionTreeBuilder Estas son las pruebas unitarias: AnyService - ExpressionTreeBuilder Unit Tests
fuente