Это упрощенная версия исходной задачи.
У меня есть класс под названием 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
Я хотел бы взять эту строку и мой экземпляр объекта и оценить ИСТИНА или ЛОЖЬ, то есть оценить Func ‹Person, bool> в экземпляре объекта.
Вот мои текущие мысли:
- Реализуйте базовую грамматику в ANTLR для поддержки базовых операторов сравнения и логических операторов. Я подумываю скопировать приоритет Visual Basic и некоторые функции здесь: http://msdn.microsoft.com/en-us/library/fw84t893(VS.80).aspx
- Попросите ANTLR создать подходящий AST из предоставленной строки.
- Пройдите по AST и используйте структуру Predicate Builder для динамического создания Func ‹Person, bool>
- При необходимости оцените предикат для экземпляра Person
У меня вопрос: не перепекли ли я это полностью? какие-нибудь альтернативы?
РЕДАКТИРОВАТЬ: выбранное решение
Я решил использовать динамическую библиотеку Linq, в частности класс динамических запросов, представленный в 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, в данном случае - ИСТИНА.
Большое спасибо Марку Гравеллу.
Включите пакет Nuget System.Linq.Dynamic, документацию здесь
CSharpSamples\LinqSamples\DynamicQuery\DynamicQuery
и затем либо 1) скопируйте классdynamic.cs
из образца проекта в свой проект, либо 2) поместитеdynamic.cs
в свой собственный проект как отдельную dll с именем файла и пространством имен по умолчаниюSystem.Linq.Dynamic
(что я и сделал, потому что он более СУХИЙ. - person toddmo   schedule 04.03.2016