Как использовать делегата для получения по имени CompiledQuery

Я пытаюсь найти и запустить CompiledQuery с именем. Как получить доступ к скомпилированному запросу по имени и как мне затем вызвать делегата?

Вот, насколько я могу получить - я получаю ошибку "Ошибка привязки к целевому методу"

public class ActivityRepository
{
    private readonly ActivityDataContext _db;

    public ActivityRepository()
    {
        _db = new ActivityDataContext();
    }

    public static Func<ActivityDataContext, int, IQueryable<ProjectObject>>
        GetCompiledLatestProjects = CompiledQuery.Compile
            ((ActivityDataContext db, int projectId) =>
             from c in db.projectObjects
             where c.projectId == projectId
             select c);

    public static Func<ActivityDataContext, Guid, IQueryable<Report>>
        GetCompiledReports = CompiledQuery.Compile
            ((ActivityDataContext db, Guid itemId) =>
             from c in db.Reports
             where c.reportObjectId == itemId
             select c);

// Другие скомпилированные запросы пропущены, но результаты - это объекты IQueryable, которые реализуют общий интерфейс IProjectObject

    delegate IQueryable<IProjectObject> MyDelegate();

    static MyDelegate GetByName(object target, string methodName)
    {
            return (MyDelegate)Delegate.CreateDelegate
                (typeof(MyDelegate), target, methodName);
    }

    public IList<Results> GetResults(string reportName)
    {
            IQueryable<ProjectObject> projectItems = GetLatestProjectObjects(projectId, quantity);
        foreach (projectObject o in projectItems)
        {
            MyDelegate del = GetByName(this, reportName);
             var dbReport = (IProjectObject) GetCompiledReports(_db, o.itemId).FirstOrDefault();
// add results to List and return
         }
     }
}

1 ответ

Решение

Вероятно, вы можете сделать какой-то долгий способ определения любых полей типа Func<...> с DataContext в качестве первого аргумента, возвращающего IQueryable, но вам может оказаться гораздо проще просто добавить пользовательский атрибут:

[AttributeUsage(AttributeTargets.Field, AllowMultiple = false)]
public class DynamicQueryAttribute : Attribute { }

... которые вы можете добавить к своему полю:

[DynamicQuery]
public static Func<ActivityDataContext, int, IQueryable<ProjectObject>>
    GetCompiledLatestProjects = CompiledQuery.Compile
        ((ActivityDataContext db, int projectId) =>
         from c in db.projectObjects
         where c.projectId == projectId
         select c);

Затем вы можете выбрать запрос на основе этого и его имя:

public static IQueryable<IProjectObject> ExecuteQuery(Type ownerType, string name, params object[] args)
{
    var query = typeof(ownerType)
        .GetFields(BindingFlags.Public | BindingFlags.Static)
        .Where(f =>
                (f.GetCustomAttributes(typeof(DynamicQueryAttribute), false).Length > 0)
                && (f.Name.ToLowerInvariant() == name.ToLowerInvariant()))
        .Select(f => (Delegate) f.GetValue(null))
        .SingleOrDefault();

    if (query == null)
        return null;

    return (IQueryable<IReportObject>)query.DynamicInvoke(args);
}

Использование:

var results = ExecuteQuery(
    typeof(ActivityRepository),
    "GetCompiledLatestProjects", 
    dataContext, 
    projectId);

Надеюсь, это поможет:)

Другие вопросы по тегам