Це спрощена версія вихідної проблеми.
У мене клас під назвою Person:
public class Person {
public string Name { get; set; }
public int Age { get; set; }
public int Weight { get; set; }
public DateTime FavouriteDay { get; set; }
}
... і скажемо про примірник:
var bob = new Person {
Name = "Bob",
Age = 30,
Weight = 213,
FavouriteDay = '1/1/2000'
}
Я хотів би записати наступне як рядок у своєму улюбленому текстовому редакторі ....
(Person.Age > 3 AND Person.Weight > 50) OR Person.Age < 3
Я хотів би взяти цей рядок та мій екземпляр об'єкта та оцінити ІСТИНУ чи ЛОЖУ - тобто оцінити функцію <Person, bool> на екземплярі об'єкта.
Ось мої поточні думки:
- Реалізуйте базову граматику в ANTLR для підтримки основних операторів порівняння та логіки. Я думаю про копіювання пріоритету Visual Basic та деяких наборів функцій тут: http://msdn.microsoft.com/en-us/library/fw84t893(VS.80).aspx
- Запропонуйте ANTLR створити відповідний AST з наданого рядка.
- Пройдіть по AST та використовуйте рамку Builder предикатів, щоб динамічно створювати функцію <Person, bool>
- Оцініть присудок щодо екземпляра Person за необхідністю
Моє запитання: чи я це повністю переоцінив? будь-які альтернативи?
EDIT: обране рішення
Я вирішив використати бібліотеку Dynamic Linq, зокрема клас Dynamic Query, наданий у LINQSamples.
Код нижче:
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();
}
}
}
Результат має тип System.Boolean, і в цьому випадку є TRUE.
Велике спасибі Марку Гравеллу.
Увімкніть System.Linq.Dynamic NuGet пакет, документація тут