Aliens 2018-10-19 13:59 采纳率: 33.3%
浏览 3062
已采纳

c# list如何实现动态分组?

目前我只会对列表的固定的多个字段进行分组,能否实现让用户选择多个字段进行分组?不要使用条件列举所有选择情况,字段数很多,组合出的数量也很大,没有操作性。
代码如下,RuleType, RuleLevel是两个分组字段,如何动态指定其他字段?使用表达式树构造lambda表达式是否可行?如何实现呢?谢谢!

StatisticList = DataList.GroupBy(t => new { t.RuleType, t.RuleLevel})
.Select(g => new Sta{
Type = g.Key.RuleType,
Level = g.Key.RuleLevel,
ScoreTop = g.Sum(t => t.ScoreTop),
}).ToList();

  • 写回答

2条回答 默认 最新

  • threenewbee 2018-10-19 17:21
    关注

    帮你写了一个
    图片说明

    完整代码如下

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Text;
    
    namespace Q703207
    {
        static class LinqExt
        {
            public class DGroupBy<T> : IGrouping<object[], T>
            {
                private List<T> _innerlist = new List<T>();
    
                private object[] _key;
    
                public DGroupBy(object[] key) { _key = key; }
    
                public object[] Key
                {
                    get { return _key; }
                }
    
                public void Add(T value)
                {
                    _innerlist.Add(value);
                }
    
                public IEnumerator<T> GetEnumerator()
                {
                    return this._innerlist.GetEnumerator();
                }
    
                System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
                {
                    return this._innerlist.GetEnumerator();
                }
            }
    
            public static IEnumerable<IGrouping<object[], T>> DynamicGroupBy<T>(this IEnumerable<T> data, string[] keys)
            {
                List<DGroupBy<T>> list = new List<DGroupBy<T>>();
                foreach (var item in data.Select(x => new { 
                    k = keys.Select(y => x.GetType().GetProperty(y).GetValue(x, null)).ToArray(),
                    v = x
                }))
                {
                    DGroupBy<T> existing = list.SingleOrDefault(x => x.Key.Zip(item.k, (a, b) => a.Equals(b)).All(y => y));
                    if (existing == null)
                    {
                        existing = new DGroupBy<T>(item.k);
                        list.Add(existing);
                    }
                    existing.Add(item.v);
                }
                return list;
            }
        }
    
        class User
        {
            public int ID { get; set; }
            public string Name { get; set; }
            public int Age { get; set; }
            public string City { get; set; }
            public override string ToString()
            {
                return string.Format("{0},{1},{2},{3}", ID, Name, Age ,City);
            }
        }
    
        class Program
        {
            static void Main(string[] args)
            {
                List<User> users = new List<User>()
                {
                    new User() { ID = 1, Age = 20, City = "BJ", Name = "A" },
                    new User() { ID = 2, Age = 20, City = "BJ", Name = "B" },
                    new User() { ID = 3, Age = 20, City = "BJ", Name = "C" },
                    new User() { ID = 4, Age = 20, City = "SH", Name = "D" },
                    new User() { ID = 5, Age = 30, City = "SH", Name = "E" },
                    new User() { ID = 6, Age = 30, City = "SH", Name = "F" },
                    new User() { ID = 7, Age = 30, City = "CD", Name = "G" },
                    new User() { ID = 8, Age = 30, City = "CD", Name = "H" },
                    new User() { ID = 9, Age = 30, City = "HK", Name = "I" },
                    new User() { ID = 10, Age = 30, City = "HK", Name = "J" },
                };
                var query = users.DynamicGroupBy(new string[] { "City" });
                foreach (var item in query)
                {
                    Console.WriteLine("Key: {0}", string.Join(",", item.Key.Select(x => x.ToString())));
                    foreach (var item1 in item)
                        Console.WriteLine("\t" + item1);
                }
            }
        }
    }
    
    

    因为csdn的女产品瞎搞,不得已,随便再问两个C#问题采纳我的回答,帮你用表达式树优化下代码(用反射比较慢)。

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论
查看更多回答(1条)

报告相同问题?

悬赏问题

  • ¥500 高有偿提问!求优化设计微信小程序
  • ¥15 matlab在安装时报错 无法找到入口 无法定位程序输入点
  • ¥15 收益高的广告联盟有哪些
  • ¥15 Android Studio webview 的使用问题, 播放器横屏全屏
  • ¥15 删掉jdk后重新下载,Java web所需要的eclipse无法使用
  • ¥15 uniapp正式环境中通过webapi将本地数据推送到设备出现的跨域问题
  • ¥15 xui建立节点,显示错误
  • ¥15 关于#单片机#的问题:开始、复位、十进制的功能可以实现,但是切换八进制的功能无法实现(按下按键也没有效果),把初始状态调成八进制,也是八进制可以实现但是切换到十进制不行(相关搜索:汇编语言|计数器)
  • ¥15 VINS-Mono或Fusion中feature_manager中estimated_depth是特征的深度还是逆深度?
  • ¥15 谷歌浏览器如何备份抖音网页数据