表达式树组并使用任何类型的列进行选择

时间:2019-03-22 17:16:43

标签: c# linq lambda expression-trees

我正在使用下面出色线程中的代码变体,使用自定义列列表从任何给定数据源动态构建树视图结构。

Expression.Call GroupBy then Select and Count()?

我的代码是此解决方案中提供的代码的递归变体:

public static List<MenuItem> GroupBySelector<TSource>(List<TSource> source, List<string> columns, int entry)
{
    string column = columns[entry];

    IQueryable<TSource> query = source.AsQueryable();

    int nextEntry = entry + 1;

    if (source == null) throw new ArgumentNullException("source");
    if (column == null) throw new ArgumentNullException("column");
    if (column.Length == 0) throw new ArgumentException("column");

    var param = Expression.Parameter(typeof(TSource));
    var prop = Expression.Property(param, column);

    Expression<Func<TSource, string>> keySelector = Expression.Lambda<Func<TSource, string>>
    (
        prop,
        param
    );

    if (columns.Count == nextEntry)
        return query.GroupBy(keySelector).Select(p => new MenuItem { Key = p.Key, Count = p.Count() }).ToList();
    else
        return query.GroupBy(keySelector)
          .Select(p => new MenuItem { Key = p.Key, Count = p.Count(), Items = GroupBySelector<TSource>(p.ToList(), columns, nextEntry) }).ToList();
}

这对于字符串类型的列非常有效;我将如何使其动态支持任何(原始)类型的列?包括包含不同类型的列的列表,例如Name(string),Age(int)等?

1 个答案:

答案 0 :(得分:0)

嗯,虽然不太明显,但这是可能的。一旦开始使用运行时类型信息,它就会向外传播,直到您在运行时编译代码为止,幸运的是,使用Expression并不太困难,但是绝对可以在C#中进行改进。

这是我的代码:

public static class MenuItemTree {
    public static MethodInfo GetGenericMethod(this Type type, string name, Type[] generic_type_args, Type[] param_types) {
        foreach (MethodInfo m in type.GetMethods())
            if (m.Name == name) {
                var pa = m.GetParameters();
                if (pa.Length == param_types.Length) {
                    var c = m.MakeGenericMethod(generic_type_args);
                    if (c.GetParameters().Select(p => p.ParameterType).SequenceEqual(param_types))
                        return c;
                }
            }
        return null;
    }

    static MethodInfo GetGroupByMethodStatically<TElement, TKey>()
        => typeof(Enumerable).GetGenericMethod("GroupBy", new[] { typeof(TElement), typeof(TKey) }, new[] { typeof(IEnumerable<TElement>), typeof(Func<TElement, TKey>) });

    static MethodInfo GetEnumerableMethod(string methodName, Type TTElement, Type TTKey) {
        var TIE = typeof(IEnumerable<>).MakeGenericType(TTElement);
        var TF = typeof(Func<,>).MakeGenericType(TTElement, TTKey);
        return typeof(Enumerable).GetGenericMethod(methodName, new[] { TTElement, TTKey }, new[] { TIE, TF });
    }

    static MethodInfo GetEnumerableMethod(string methodName, Type TTElement) {
        var TIE = typeof(IEnumerable<>).MakeGenericType(TTElement);
        return typeof(Enumerable).GetGenericMethod(methodName, new[] { TTElement }, new[] { TIE });
    }

    public static List<MenuItem> GroupBySelector<TElement>(IEnumerable<TElement> source, IList<string> columnNames, int entry = 0) {
        if (source == null) throw new ArgumentNullException(nameof(source));

        string columnName = columnNames[entry];

        if (columnName == null) throw new ArgumentNullException(nameof(columnName));
        if (columnName.Length == 0) throw new ArgumentException(nameof(columnName));

        int nextEntry = entry + 1;

        var TTElement = typeof(TElement);
        var TIETElement = typeof(IEnumerable<TElement>);

        var keyParm = Expression.Parameter(TTElement);
        var prop = Expression.Property(keyParm, columnName);

        var parm = Expression.Parameter(TIETElement, "p");
        var gbmi = GetEnumerableMethod("GroupBy", TTElement, prop.Type);
        var gbExpr = Expression.Lambda(prop, keyParm);
        var body = Expression.Call(gbmi, parm, gbExpr);

        var TSelectInput = typeof(IGrouping<,>).MakeGenericType(prop.Type, TTElement);
        var separm = Expression.Parameter(TSelectInput, "p");

        var miKey = typeof(MenuItem).GetMember("Key").Single();
        var miCount = typeof(MenuItem).GetMember("Count").Single();
        var miItems = typeof(MenuItem).GetMember("Items").Single();

        Expression sepKey = Expression.Property(separm, "Key");
        if (sepKey.Type != typeof(string)) {
            var ctmi = sepKey.Type.GetMethod("ToString", Type.EmptyTypes);
            sepKey = Expression.Call(sepKey, ctmi);
        }
        var countmi = GetEnumerableMethod("Count", TTElement);
        var sepCount = Expression.Call(countmi, separm);

        var gbsmi = GetGenericMethod(MethodBase.GetCurrentMethod().DeclaringType, "GroupBySelector", new[] { TTElement }, new[] { TIETElement, typeof(IList<string>), typeof(int) });
        var sepItems = Expression.Call(gbsmi, separm, Expression.Constant(columnNames), Expression.Constant(nextEntry));

        var senew = Expression.New(typeof(MenuItem));

        var se1body = Expression.MemberInit(senew, new[] {
                        Expression.Bind(miKey, sepKey),
                        Expression.Bind(miCount, sepCount),
                        Expression.Bind(miItems, sepItems)
                    });
        var se1 = Expression.Lambda(se1body, separm);

        var se2body = Expression.MemberInit(senew, new[] {
                        Expression.Bind(miKey, sepKey),
                        Expression.Bind(miCount, sepCount)
                    });
        var se2 = Expression.Lambda(se2body, separm);

        var smi = GetEnumerableMethod("Select", TSelectInput, typeof(MenuItem));
        body = Expression.Call(smi, body, (nextEntry < columnNames.Count) ? se1: se2);

        var lmi = GetEnumerableMethod("ToList", typeof(MenuItem));
        body = Expression.Call(lmi, body);

        var returnFunc = Expression.Lambda<Func<IEnumerable<TElement>, List<MenuItem>>>(body, parm).Compile();
        return returnFunc(source);
    }
}