zoukankan      html  css  js  c++  java
  • List去重

    Enumerable.Distinct 方法 是常用的LINQ扩展方法,属于System.Linq的Enumerable方法,可用于去除数组、集合中的重复元素,还可以自定义去重的规则。

    有两个重载方法:

    复制代码
    复制代码
            //
            // 摘要: 
            //     通过使用默认的相等比较器对值进行比较返回序列中的非重复元素。
            //
            // 参数: 
            //   source:
            //     要从中移除重复元素的序列。
            //
            // 类型参数: 
            //   TSource:
            //     source 中的元素的类型。
            //
            // 返回结果: 
            //     一个 System.Collections.Generic.IEnumerable<T>,包含源序列中的非重复元素。
            //
            // 异常: 
            //   System.ArgumentNullException:
            //     source 为 null。
            public static IEnumerable<TSource> Distinct<TSource>(this IEnumerable<TSource> source);
            //
            // 摘要: 
            //     通过使用指定的 System.Collections.Generic.IEqualityComparer<T> 对值进行比较返回序列中的非重复元素。
            //
            // 参数: 
            //   source:
            //     要从中移除重复元素的序列。
            //
            //   comparer:
            //     用于比较值的 System.Collections.Generic.IEqualityComparer<T>。
            //
            // 类型参数: 
            //   TSource:
            //     source 中的元素的类型。
            //
            // 返回结果: 
            //     一个 System.Collections.Generic.IEnumerable<T>,包含源序列中的非重复元素。
            //
            // 异常: 
            //   System.ArgumentNullException:
            //     source 为 null。
            public static IEnumerable<TSource> Distinct<TSource>(this IEnumerable<TSource> source, IEqualityComparer<TSource> comparer);    
    复制代码
    复制代码

    第一个方法不带参数,第二个方法需要传一个System.Collections.Generic.IEqualityComparer<T>的实现对象

    1.值类型元素集合去重

    List<int> list = new List<int> { 1, 1, 2, 2, 3, 4, 5, 5 };
    list.Distinct().ToList().ForEach(s => Console.WriteLine(s));

    执行结果是:1 2 3 4 5

    2.引用类型元素集合去重

    首先自定义一个Student类

    使用不到参数的Distinct方法去重

    复制代码
    复制代码
                List<Student> list = new List<Student>() { 
                    new Student("James",1,"Basketball"),
                    new Student("James",1,"Basketball"),
                    new Student("Kobe",2,"Basketball"),
                    new Student("Curry",3,"Football"),
                    new Student("Curry",3,"Yoga")
                };
                list.Distinct().ToList().ForEach(s => Console.WriteLine(s.ToString()));   
    复制代码
    复制代码

    执行结果:

    可见,并没有去除重复的记录。

    不带comparer参数的Distinct方法是使用的IEqualityComparer接口的默认比较器进行比较的,对于引用类型,默认比较器比较的是其引用地址,程序中集合里的每一个元素都是个新的实例,引用地址都是不同的,所以不会被作为重复记录删除掉。

    因此,我们考虑使用第二个重载方法。

    新建一个类,实现IEqualityComparer接口。注意GetHashCode方法的实现,只有HashCode相同才会去比较

    复制代码
    复制代码
        public class Compare:IEqualityComparer<Student>
        {
            public bool Equals(Student x,Student y)
            {
                return x.Id == y.Id;//可以自定义去重规则,此处将Id相同的就作为重复记录,不管学生的爱好是什么
            }
            public int GetHashCode(Student obj)
            {
                return obj.Id.GetHashCode();
            }
        }
    复制代码
    复制代码

    然后调用

    list.Distinct(new Compare()).ToList().ForEach(s => Console.WriteLine(s.ToString()));

    执行结果:

    我们按照Id去给这个集合去重成功!

    3.如何编写一个具有扩展性的去重方法

    复制代码
        public class Compare<T, C> : IEqualityComparer<T>
        {
            private Func<T, C> _getField;
            public Compare(Func<T, C> getfield)
            {
                this._getField = getfield;
            }
            public bool Equals(T x, T y)
            {
                return EqualityComparer<C>.Default.Equals(_getField(x), _getField(y));
            }
            public int GetHashCode(T obj)
            {
                return EqualityComparer<C>.Default.GetHashCode(this._getField(obj));
            }
        }
        public static class CommonHelper
        {
            /// <summary>
            /// 自定义Distinct扩展方法
            /// </summary>
            /// <typeparam name="T">要去重的对象类</typeparam>
            /// <typeparam name="C">自定义去重的字段类型</typeparam>
            /// <param name="source">要去重的对象</param>
            /// <param name="getfield">获取自定义去重字段的委托</param>
            /// <returns></returns>
            public static IEnumerable<T> MyDistinct<T, C>(this IEnumerable<T> source, Func<T, C> getfield)
            {
                return source.Distinct(new Compare<T, C>(getfield));
            }
        }
    复制代码

    调用:

    list.MyDistinct(s=>s.Id).ToList().ForEach(s => Console.WriteLine(s.ToString()));

    用到了泛型、委托、扩展方法等知识点。可以用于任何集合的各种去重场景

  • 相关阅读:
    iOS 版 MWeb 发布到自建 Wordpress 和 Metaweblog API 使用指南
    MWeb for iOS 测试版介绍
    Contributing to the C++ Core Guidelines
    MWeb 2.0.7 版发布!
    MWeb 2.0 测试版可以下载啦,这次是公开测试,感兴趣的朋友可以试试
    MWeb 1.7.1 版发布!支持导出为 RTF 和 Docx、发布到 Evernote 带样式、文档库备份和新网站主题等大量改进!
    《苏州爱情故事》
    把 MWeb Lite 的文档库文档和数据搬到 MWeb 正式版中
    MWeb 1.6 发布!Dark Mode、全文搜寻、发布到Wordpress、Evernote 等支持更新、编辑/预览视图模式等
    MWeb 1.5 发布!增加打字机滚动模式、发布到 Evernote、印象笔记、Wordpress.com、Blogger、编辑器内代码块语法高亮
  • 原文地址:https://www.cnblogs.com/ztf20/p/9117844.html
Copyright © 2011-2022 走看看