爆款云主机2核4G限时秒杀,88元/年起!
查看详情

活动

天翼云最新优惠活动,涵盖免费试用,产品折扣等,助您降本增效!
热门活动
  • 618智算钜惠季 爆款云主机2核4G限时秒杀,88元/年起!
  • 免费体验DeepSeek,上天翼云息壤 NEW 新老用户均可免费体验2500万Tokens,限时两周
  • 云上钜惠 HOT 爆款云主机全场特惠,更有万元锦鲤券等你来领!
  • 算力套餐 HOT 让算力触手可及
  • 天翼云脑AOne NEW 连接、保护、办公,All-in-One!
  • 中小企业应用上云专场 产品组合下单即享折上9折起,助力企业快速上云
  • 息壤高校钜惠活动 NEW 天翼云息壤杯高校AI大赛,数款产品享受线上订购超值特惠
  • 天翼云电脑专场 HOT 移动办公新选择,爆款4核8G畅享1年3.5折起,快来抢购!
  • 天翼云奖励推广计划 加入成为云推官,推荐新用户注册下单得现金奖励
免费活动
  • 免费试用中心 HOT 多款云产品免费试用,快来开启云上之旅
  • 天翼云用户体验官 NEW 您的洞察,重塑科技边界

智算服务

打造统一的产品能力,实现算网调度、训练推理、技术架构、资源管理一体化智算服务
智算云(DeepSeek专区)
科研助手
  • 算力商城
  • 应用商城
  • 开发机
  • 并行计算
算力互联调度平台
  • 应用市场
  • 算力市场
  • 算力调度推荐
一站式智算服务平台
  • 模型广场
  • 体验中心
  • 服务接入
智算一体机
  • 智算一体机
大模型
  • DeepSeek-R1-昇腾版(671B)
  • DeepSeek-R1-英伟达版(671B)
  • DeepSeek-V3-昇腾版(671B)
  • DeepSeek-R1-Distill-Llama-70B
  • DeepSeek-R1-Distill-Qwen-32B
  • Qwen2-72B-Instruct
  • StableDiffusion-V2.1
  • TeleChat-12B

应用商城

天翼云精选行业优秀合作伙伴及千余款商品,提供一站式云上应用服务
进入甄选商城进入云市场创新解决方案
办公协同
  • WPS云文档
  • 安全邮箱
  • EMM手机管家
  • 智能商业平台
财务管理
  • 工资条
  • 税务风控云
企业应用
  • 翼信息化运维服务
  • 翼视频云归档解决方案
工业能源
  • 智慧工厂_生产流程管理解决方案
  • 智慧工地
建站工具
  • SSL证书
  • 新域名服务
网络工具
  • 翼云加速
灾备迁移
  • 云管家2.0
  • 翼备份
资源管理
  • 全栈混合云敏捷版(软件)
  • 全栈混合云敏捷版(一体机)
行业应用
  • 翼电子教室
  • 翼智慧显示一体化解决方案

合作伙伴

天翼云携手合作伙伴,共创云上生态,合作共赢
天翼云生态合作中心
  • 天翼云生态合作中心
天翼云渠道合作伙伴
  • 天翼云代理渠道合作伙伴
天翼云服务合作伙伴
  • 天翼云集成商交付能力认证
天翼云应用合作伙伴
  • 天翼云云市场合作伙伴
  • 天翼云甄选商城合作伙伴
天翼云技术合作伙伴
  • 天翼云OpenAPI中心
  • 天翼云EasyCoding平台
天翼云培训认证
  • 天翼云学堂
  • 天翼云市场商学院
天翼云合作计划
  • 云汇计划
天翼云东升计划
  • 适配中心
  • 东升计划
  • 适配互认证

开发者

开发者相关功能入口汇聚
技术社区
  • 专栏文章
  • 互动问答
  • 技术视频
资源与工具
  • OpenAPI中心
开放能力
  • EasyCoding敏捷开发平台
培训与认证
  • 天翼云学堂
  • 天翼云认证
魔乐社区
  • 魔乐社区

支持与服务

为您提供全方位支持与服务,全流程技术保障,助您轻松上云,安全无忧
文档与工具
  • 文档中心
  • 新手上云
  • 自助服务
  • OpenAPI中心
定价
  • 价格计算器
  • 定价策略
基础服务
  • 售前咨询
  • 在线支持
  • 在线支持
  • 工单服务
  • 建议与反馈
  • 用户体验官
  • 服务保障
  • 客户公告
  • 会员中心
增值服务
  • 红心服务
  • 首保服务
  • 客户支持计划
  • 专家技术服务
  • 备案管家

了解天翼云

天翼云秉承央企使命,致力于成为数字经济主力军,投身科技强国伟大事业,为用户提供安全、普惠云服务
品牌介绍
  • 关于天翼云
  • 智算云
  • 天翼云4.0
  • 新闻资讯
  • 天翼云APP
基础设施
  • 全球基础设施
  • 信任中心
最佳实践
  • 精选案例
  • 超级探访
  • 云杂志
  • 分析师和白皮书
  • 天翼云·创新直播间
市场活动
  • 2025智能云生态大会
  • 2024智算云生态大会
  • 2023云生态大会
  • 2022云生态大会
  • 天翼云中国行
天翼云
  • 活动
  • 智算服务
  • 产品
  • 解决方案
  • 应用商城
  • 合作伙伴
  • 开发者
  • 支持与服务
  • 了解天翼云
      • 文档
      • 控制中心
      • 备案
      • 管理中心

      List集合对象去重及按属性去重的8种方法-java基础总结系列第六篇

      首页 知识中心 数据库 文章详情页

      List集合对象去重及按属性去重的8种方法-java基础总结系列第六篇

      2023-05-23 08:22:30 阅读次数:473

      java,List,集合

      List集合对象去重及按属性去重的8种方法-java基础总结系列第六篇
      最近在写一些关于java基础的文章,但是我又不想按照教科书的方式去写知识点的文章,因为意义不大。基础知识太多了,如何将这些知识归纳总结,总结出优缺点或者是使用场景才是对知识的升华。所以我更想把java相关的基础知识进行穿针引线,进行整体上的总结。

      • 总结java中创建并写文件的5种方式
      • 总结java从文件中读取数据的6种方法
      • 总结java创建文件夹的4种方法及其优缺点
      • 总结java中删除文件或文件夹的7种方法
      • 总结java中文件拷贝剪切的5种方式

      比如之前我已经写了上面的这些内容,如果对java基础知识总结系列感兴趣的同学可以关注我的博客(文末给出我的博客地址)。

      一、本文梗概

      这一篇文章我想写一下List集合元素去重的8种方法,实际上通过灵活的运用、排列组合不一定是8种,可能有18种方法。

      • 对象元素整体去重的4种方法
      • 按照对象属性去重的4种方法

      为了在下文中进行测试内容讲解,我们先做一些初始化数据

      public class ListRmDuplicate {
        private List<String> list;
        private List<Player> playerList;
      
        @BeforeEach
        public void setup() {
          list  =  new ArrayList<>();
          list.add("kobe");
          list.add("james");
          list.add("curry");
          list.add("zimug");
          list.add("zimug");
      
          playerList= new ArrayList<>();
          playerList.add(new Player("kobe","10000"));  //科比万岁
          playerList.add(new Player("james","32"));
          playerList.add(new Player("curry","30"));
          playerList.add(new Player("zimug","27"));   // 注意这里名字重复
          playerList.add(new Player("zimug","18"));   //注意这里名字和年龄重复
          playerList.add(new Player("zimug","18")); //注意这里名字和年龄重复
      
        }
      }
      

      Player对象就是一个普通的java对象,有两个成员变量name与age,实现了带参数构造函数、toString、equals和hashCode方法、以及GET/SET方法。

      二、集合元素整体去重

      下文中四种方法对List中的String类型以集合元素对象为单位整体去重。如果你的List放入的是Object对象,需要你去实现对象的equals和hashCode方法,去重的代码实现方法和List<String>去重是一样的。

      第一种方法

      是大家最容易想到的,先把List数据放入Set,因为Set数据结构本身具有去重的功能,所以再将SET转为List之后就是去重之后的结果。这种方法在去重之后会改变原有的List元素顺序,因为HashSet本身是无序的,而TreeSet排序也不是List种元素的原有顺序。

      @Test
      void testRemove1()  {
        /*Set<String> set = new HashSet<>(list);
        List<String> newList = new ArrayList<>(set);*/
      
        //去重并排序的方法(如果是字符串,按字母表排序。如果是对象,按Comparable接口实现排序)
        //List<String> newList = new ArrayList<>(new TreeSet<>(list));
      
        //简写的方法
        List<String> newList = new ArrayList<>(new HashSet<>(list));
      
        System.out.println( "去重后的集合: " + newList);
      }
      

      控制台打印结果如下:

      去重后的集合: [kobe, james, zimug, curry]
      

      第二种方法

      使用就比较简单,先用stream方法将集合转换成流,然后distinct去重,最后在将Stream流collect收集为List。

      @Test
      void testRemove2()  {
        List<String> newList = list.stream().distinct().collect(Collectors.toList());
      
        System.out.println( "去重后的集合: " + newList);
      }
      

      控制台打印结果如下:

      去重后的集合: [kobe, james, curry, zimug]
      

      第三种方法
      这种方法利用了set.add(T),如果T元素已经存在集合中,就返回false。利用这个方法进行是否重复的数据判断,如果不重复就放入一个新的newList中,这个newList就是最终的去重结果

      //三个集合类list、newList、set,能够保证顺序
      @Test
      void testRemove3()  {
      
        Set<String> set = new HashSet<>();
        List<String> newList = new  ArrayList<>();
        for (String str :list) {
          if(set.add(str)){ //重复的话返回false
            newList.add(str);
          }
        }
        System.out.println( "去重后的集合: " + newList);
      
      }
      

      控制台打印结果和第二种方法一致。

      第四种方法
      这种方法已经脱离了使用Set集合进行去重的思维,而是使用newList.contains(T)方法,在向新的List添加数据的时候判断这个数据是否已经存在,如果存在就不添加,从而达到去重的效果。

      //优化 List、newList、set,能够保证顺序
      @Test
      void testRemove4() {
      
        List<String> newList = new  ArrayList<>();
        for (String cd:list) {
          if(!newList.contains(cd)){  //主动判断是否包含重复元素
            newList.add(cd);
          }
        }
        System.out.println( "去重后的集合: " + newList);
      
      }
      

      控制台打印结果和第二种方法一致。

      三、按照集合元素对象属性去重

      其实在实际的工作中,按照集合元素对象整体去重的应用的还比较少,更多的是要求我们按照元素对象的某些属性进行去重。
      看到这里请大家回头去看一下上文中,构造的初始化数据playerList,特别注意其中的一些重复元素,以及成员变量重复。

      第一种方法
      为TreeSet实现Comparator接口,如果我们希望按照Player的name属性进行去重,就去在Comparator接口中比较name。下文中写了两种实现Comparator接口方法:

      • lambda表达式:(o1, o2) -> o1.getName().compareTo(o2.getName())
      • 方法引用:Comparator.comparing(Player::getName)
      @Test
      void testRemove5() {
        //Set<Player> playerSet = new TreeSet<>((o1, o2) -> o1.getName().compareTo(o2.getName()));
        Set<Player> playerSet = new TreeSet<>(Comparator.comparing(Player::getName));
        playerSet.addAll(playerList);
      
        /*new ArrayList<>(playerSet).forEach(player->{
          System.out.println(player.toString());
        });*/
        //将去重之后的结果打印出来
        new ArrayList<>(playerSet).forEach(System.out::println);
      }
      

      输出结果如下:三个zimug因为name重复,另外两个被去重。但是因为使用到了TreeSet,list中元素被重新排序。

      Player{name='curry', age='30'}
      Player{name='james', age='32'}
      Player{name='kobe', age='10000'}
      Player{name='zimug', age='27'}
      

      第二种方法
      这种方法是网上很多的文章中用来显示自己很牛的方法,但是在笔者看来有点脱了裤子放屁,多此一举。既然大家都说有这种方法,我不写好像我不牛一样。我为什么说这种方法是“脱了裤子放屁”?

      • 首先用stream()把list集合转换成流
      • 然后用collect及toCollection把流转换成集合
      • 然后剩下的就和第一种方法一样了

      前两步不是脱了裤子放屁么?看看就得了,实际应用意义不大,但是如果是为了学习Stream流的使用方法,搞出这么一个例子还是有可取之处的。

      @Test
      void testRemove6() {
        List<Player> newList = playerList.stream().collect(Collectors
                .collectingAndThen(
                        Collectors.toCollection(() -> new TreeSet<>(Comparator.comparing(Player::getName))),
                        ArrayList::new));
      
        newList.forEach(System.out::println);
      }
      

      控制台打印输出和第一种方法一样。

      第三种方法

      这种方法也是笔者建议大家使用的一种方法,咋一看好像代码量更大了,但实际上这种方法是应用比较简单的方法。

      Predicate(有人管这个叫断言,从英文的角度作为名词可以翻译为谓词,作为动词可以翻译为断言)。谓词就是用来修饰主语的,比如:喜欢唱歌的小鸟,喜欢唱歌就是谓词,用来限定主语的范围。所以我们这里是用来filter过滤的,也是用来限制主语范围的,所以我认为翻译为谓词更合适。随便吧,看你怎么觉得怎么理解合理、好记,你就怎么来。

      • 首先我们定义一个谓词Predicate用来过滤,过滤的条件是distinctByKey。谓词返回ture元素保留,返回false元素被过滤掉。
      • 当然我们的需求是过滤掉重复元素。我们去重逻辑是通过map的putIfAbsent实现的。putIfAbsent方法添加键值对,如果map集合中没有该key对应的值,则直接添加,并返回null,如果已经存在对应的值,则依旧为原来的值。
      • 如果putIfAbsent返回null表示添加数据成功(不重复),如果putIfAbsent返回value(value==null :false),则满足了distinctByKey谓词的条件元素被过滤掉。

      这种方法虽然看上去代码量增大了,但是distinctByKey谓词方法只需要被定义一次,就可以无限复用。

      @Test
      void testRemove7() {
        List<Player> newList = new ArrayList<>();
        playerList.stream().filter(distinctByKey(p -> p.getName()))  //filter保留true的值
                .forEach(newList::add);
      
        newList.forEach(System.out::println);
      }
      
      static <T> Predicate<T> distinctByKey(Function<? super T, ?> keyExtractor) {
        Map<Object,Boolean> seen = new ConcurrentHashMap<>();
        //putIfAbsent方法添加键值对,如果map集合中没有该key对应的值,则直接添加,并返回null,如果已经存在对应的值,则依旧为原来的值。
        //如果返回null表示添加数据成功(不重复),不重复(null==null :TRUE)
        return t -> seen.putIfAbsent(keyExtractor.apply(t), Boolean.TRUE) == null;
      }
      

      输出结果如下:三个zimug因为name重复,另外两个被去重。并且没有打乱List的原始顺序

      Player{name='kobe', age='10000'}
      Player{name='james', age='32'}
      Player{name='curry', age='30'}
      Player{name='zimug', age='27'}
      

      第四种方法
      第四种方法实际上不是新方法,上面的例子都是按某一个对象属性进行去重,如果我们想按照某几个元素进行去重,就需要对上面的三种方法进行改造。
      我只改造其中一个,另外几个改造的原理是一样的,就是把多个比较属性加起来,作为一个String属性进行比较。

      @Test
      void testRemove8() {
        Set<Player> playerSet = new TreeSet<>(Comparator.comparing(o -> (o.getName() + "" + o.getAge())));
      
        playerSet.addAll(playerList);
      
        new ArrayList<>(playerSet).forEach(System.out::println);
      }
      版权声明:本文内容来自第三方投稿或授权转载,原文地址:https://www.cnblogs.com/zimug/p/13625424.html,作者:字母哥博客,版权归原作者所有。本网站转在其作品的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如因作品内容、版权等问题需要同本网站联系,请发邮件至ctyunbbs@chinatelecom.cn沟通。

      上一篇:在Linux中安装MySQL报错“[ERROR] --initialize specified but the data directory has files in it.“

      下一篇:MySQL报错“You must reset your password using ALTER USER statement before executing this statement.“

      相关文章

      2025-05-16 09:15:24

      Redis Set集合

      Redis Set集合

      2025-05-16 09:15:24
      set , 个数 , 元素 , 示例 , 集合
      2025-05-16 09:15:24

      jQuery遍历对象、数组、集合

      jQuery遍历对象、数组、集合

      2025-05-16 09:15:24
      jQuery , 对象 , 数组 , 遍历 , 集合
      2025-05-14 10:02:58

      java项目多端数据同步解决方案

      多端数据同步是指在多个设备(例如桌面应用、移动应用、Web应用)之间保持数据的一致性。

      2025-05-14 10:02:58
      java , Spring , WebSocket , 同步 , 数据 , 版本号
      2025-05-14 10:02:58

      java休眠到指定时间怎么写

      java休眠到指定时间怎么写

      2025-05-14 10:02:58
      java , sleep , Thread , util , 方法
      2025-05-13 09:53:23

      Java交换map的key和value值

      在Java中,我们都知道直接交换Map的key和value是不被允许的,因为Map的接口设计是基于key-value对的,其中key是唯一的,并且是不可变的(在HashMap等常见的实现中,虽然key的引用是不可变的,但key对象本身如果是可变的,它的内容是可以变化的,但这样做可能会导致不正确的行为或异常)。

      2025-05-13 09:53:23
      key , List , Map , null , value , 映射 , 键值
      2025-05-13 09:49:12

      Java学习(动态代理的思想详细分析与案例准备)(1)

      Java学习(动态代理的思想详细分析与案例准备)(1)

      2025-05-13 09:49:12
      java , 代理 , 代码 , 对象 , 接口 , 方法 , 需要
      2025-05-12 08:40:18

      List<T>的使用

      List<T>的使用

      2025-05-12 08:40:18
      List , string
      2025-05-09 08:50:35

      DS进阶:并查集

      在一些应用问题中,需要将n个不同的元素划分成一些不相交的集合。开始时,每个元素自成一个单元素集合,然后按一定的规律将归于同一组元素的集合合并。在此过程中要反复用到查询某一个元素归属于那个集合的运算。适合于描述这类问题的抽象数据类型称为并查集(union-find set)。

      2025-05-09 08:50:35
      一个 , 元素 , 查找 , 查集 , 节点 , 集合
      2025-05-09 08:20:32

      基于IDEA的Maven简单工程创建及结构分析

      通过一个 mvn 命令直接让我们创建一个 Maven 的脚手架。

      2025-05-09 08:20:32
      java , Maven , xml , 创建 , 文件 , 文件夹 , 项目
      2025-05-08 09:03:57

      前K个高频元素java

      给定一个非空的整数数组,返回其中出现频率前 前K个高频元素java 高的元素。

      2025-05-08 09:03:57
      java , 元素 , 样例 , 给定
      查看更多
      推荐标签

      作者介绍

      天翼云小翼
      天翼云用户

      文章

      33561

      阅读量

      5226078

      查看更多

      最新文章

      线性表练习之Example038-编写一个函数将链表 h2 链接到链表 h1 之后,要求链接后的链表仍然保持循环链表形式

      2025-04-14 09:26:51

      java使用JDBC方式操作mysql数据库示例

      2025-04-11 07:15:54

      【mongodb基础-3】文档/集合、数据库等基础概念、mongodb数据类型的介绍

      2025-03-11 09:35:39

      MongoDB从0到1:高效数据使用方法

      2025-03-03 09:36:20

      数据结构之并查集

      2025-02-11 09:39:39

      数据结构之ArrayList与顺序表(下)

      2025-02-11 09:38:28

      查看更多

      热门文章

      关于PyTorch继承nn.Module出现raise NotImplementedError的问题解决方案

      2023-02-27 10:10:19

      取出一个实体中不为null的属性和属性值

      2022-12-29 09:29:46

      字符输入流一个一个读数据

      2023-03-30 09:22:32

      解决Hbase报错java.lang.IllegalStateException: The procedure WAL relies on the ability to hsync for....

      2023-04-19 09:38:35

      redis-数据类型-普通集合

      2023-03-29 10:07:52

      字符串截取

      2023-02-24 08:40:10

      查看更多

      热门标签

      数据库 mysql 字符串 数据结构 MySQL 算法 redis oracle java sql python 数据 索引 SQL 查询
      查看更多

      相关产品

      弹性云主机

      随时自助获取、弹性伸缩的云服务器资源

      天翼云电脑(公众版)

      便捷、安全、高效的云电脑服务

      对象存储

      高品质、低成本的云上存储服务

      云硬盘

      为云上计算资源提供持久性块存储

      查看更多

      随机文章

      基础数据类型说明

      求煤球的数量

      Mybatis if test 判断 list不为 null 并且判断集合大小不为0

      Lc3-无重复字符的最长子串

      用java写个redis工具类

      生成当天递增唯一的流水号的几种方式

      • 7*24小时售后
      • 无忧退款
      • 免费备案
      • 专家服务
      售前咨询热线
      400-810-9889转1
      关注天翼云
      • 旗舰店
      • 天翼云APP
      • 天翼云微信公众号
      服务与支持
      • 备案中心
      • 售前咨询
      • 智能客服
      • 自助服务
      • 工单管理
      • 客户公告
      • 涉诈举报
      账户管理
      • 管理中心
      • 订单管理
      • 余额管理
      • 发票管理
      • 充值汇款
      • 续费管理
      快速入口
      • 天翼云旗舰店
      • 文档中心
      • 最新活动
      • 免费试用
      • 信任中心
      • 天翼云学堂
      云网生态
      • 甄选商城
      • 渠道合作
      • 云市场合作
      了解天翼云
      • 关于天翼云
      • 天翼云APP
      • 服务案例
      • 新闻资讯
      • 联系我们
      热门产品
      • 云电脑
      • 弹性云主机
      • 云电脑政企版
      • 天翼云手机
      • 云数据库
      • 对象存储
      • 云硬盘
      • Web应用防火墙
      • 服务器安全卫士
      • CDN加速
      热门推荐
      • 云服务备份
      • 边缘安全加速平台
      • 全站加速
      • 安全加速
      • 云服务器
      • 云主机
      • 智能边缘云
      • 应用编排服务
      • 微服务引擎
      • 共享流量包
      更多推荐
      • web应用防火墙
      • 密钥管理
      • 等保咨询
      • 安全专区
      • 应用运维管理
      • 云日志服务
      • 文档数据库服务
      • 云搜索服务
      • 数据湖探索
      • 数据仓库服务
      友情链接
      • 中国电信集团
      • 189邮箱
      • 天翼企业云盘
      • 天翼云盘
      ©2025 天翼云科技有限公司版权所有 增值电信业务经营许可证A2.B1.B2-20090001
      公司地址:北京市东城区青龙胡同甲1号、3号2幢2层205-32室
      • 用户协议
      • 隐私政策
      • 个人信息保护
      • 法律声明
      备案 京公网安备11010802043424号 京ICP备 2021034386号