объединить метод MethodCallExpression для MemberExpression - PullRequest
0 голосов
/ 25 февраля 2019

Я хочу создать следующее выражение Linq, используя дерево выражений:

var result2 = soldiers.OrderByDescending(soldier => soldier.Ranks.OrderBy(r=> r.Date).FirstOrDefault().Date).ToArray() ;

Модели перечислены в конце вопроса

Я пытался сделать следующее:

ParameterExpression pe = Expression.Parameter(typeof(Soldier), "soldier");

// e.g soldier.ranks
MemberExpression rank = Expression.Property(pe, "ranks");

ParameterExpression nestedParams = Expression.Parameter(typeof(Rank), "rank");

var dateProperty = typeof(Rank).GetProperty("Date");

// e.g rank.Date
MemberExpression nestedExpression= Expression.MakeMemberAccess(nestedParams, dateProperty);

// e.g rank => rank.date
var orderByExp = Expression.Lambda<Func<Rank, DateTime?>>(nestedExpression, nestedParams);

MethodCallExpression orderByCallExpression2 = Expression.Call(
typeof(Queryable),
"OrderByDescending",
 new Type[] { typeof(Rank), typeof(DateTime?) },
 rank, ***// i suspect the problem is in this line #110***
 orderByExp);

Но возникла следующая ошибка:

System.InvalidOperationException: 'No generic method 'OrderByDescending' on type 'System.Linq.Queryable' is compatible with the supplied type arguments and arguments. No type arguments should be provided if the method is non-generic. '

Как я уже писал в комментариях к коду, я подозреваю, что проблема возникает из-застроки 110. Я попытался объединить метод orderBy со свойством soldier.ranks.Но, возможно, из-за того, что 'soldier.ranks - это MemberExpression вместо MethodCallExpression или Queryable, я не смог предварительно выполнить это действие.

Я использовал обе ссылки в качестве ссылки, но смог найти что-то похожее на мою проблему.* https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/concepts/expression-trees/how-to-use-expression-trees-to-build-dynamic-queries

*https://stackoverflow.com/a/11337472/10950549

модели:

public class Soldier
{
    public Rank Rank {
        get {
            return this.Ranks.OrderByDescending(m => m.Date).FirstOrDefault();
        }
    }

    public ICollection<Rank> Ranks { get; set; }
}

public class Rank
{
    public int Id { get; set; }

    public DateTime? Date { get; set; }
}

спасибо,

1 Ответ

0 голосов
/ 26 февраля 2019

Я подозреваю, что проблема возникает из-за строки 110

MethodCallExpression orderByCallExpression2 = Expression.Call(
typeof(Queryable),
"OrderByDescending",
 new Type[] { typeof(Rank), typeof(DateTime?) },
 rank, ***// i suspect the problem is in this line #110***
 orderByExp);

Нет, проблема в Queryable 3 строки выше:

typeof(Queryable),

Неважно, какой именно тип выражения (член, вызов метода и т. Д.) rank, важно то, каков тип результата этого выражения (т. Е. Expression.Type).И тип soldier.Ranks является ICollection<Rank> - определенно не IQueryable<Rank>, следовательно, исключение.

Но, как мы знаем, ICollection<T> наследует (является) IEnumerable<T>, следовательно, oldier.Ranks.OrderBy на самом деле является вызовом Enumerable.OrderBy, а не Queryable.OrderBy.

С учетом сказанногопросто измените Queryable на Enumerable в вышеупомянутой строке, и эта конкретная проблема будет решена.

...