当前位置: 首页 > article >正文

Java retainAll() 详解

在 Java 中,retainAll()Collection 接口(ListSet 等集合类实现该接口)的一种方法,用于保留集合中与指定集合交集的元素,删除其他所有元素。

以下是对 retainAll() 方法的详细讲解。


1. 方法定义

方法签名

boolean retainAll(Collection<?> c)

参数

  • c:一个集合,用于指定要保留的元素。

返回值

  • 返回一个 boolean 值:
    • true:如果集合内容因调用此方法而改变。
    • false:如果集合内容没有改变(即调用此方法前后集合中的元素相同)。

2. 功能描述

  • retainAll() 方法会将调用该方法的集合(假设为 A)中的所有元素与参数集合(假设为 B)进行比较,保留两者交集的元素。
  • 如果 A 中的元素不在 B 中,它们会被移除。
  • 参数集合 B 不会被修改。

3. 使用示例

基本用法

import java.util.ArrayList;
import java.util.Arrays;

public class RetainAllExample {
    public static void main(String[] args) {
        ArrayList<Integer> list1 = new ArrayList<>(Arrays.asList(1, 2, 3, 4, 5));
        ArrayList<Integer> list2 = new ArrayList<>(Arrays.asList(3, 4, 5, 6, 7));

        // 保留 list1 和 list2 的交集
        list1.retainAll(list2);

        System.out.println(list1); // 输出:[3, 4, 5]
    }
}

返回值示例

import java.util.ArrayList;
import java.util.Arrays;

public class RetainAllExample {
    public static void main(String[] args) {
        ArrayList<String> list1 = new ArrayList<>(Arrays.asList("A", "B", "C"));
        ArrayList<String> list2 = new ArrayList<>(Arrays.asList("B", "C"));

        boolean isModified = list1.retainAll(list2);

        System.out.println(isModified); // 输出:true,因为 list1 发生了改变
        System.out.println(list1); // 输出:[B, C]
    }
}

4. 常见场景

场景 1:找出两个集合的交集

使用 retainAll() 可以快速找到两个集合的交集:

Set<Integer> set1 = new HashSet<>(Arrays.asList(1, 2, 3, 4));
Set<Integer> set2 = new HashSet<>(Arrays.asList(3, 4, 5, 6));

set1.retainAll(set2);

System.out.println(set1); // 输出:[3, 4]

场景 2:从集合中删除不需要的元素

通过与一个已知集合比较,保留需要的元素,其余元素被删除:

List<String> fruits = new ArrayList<>(Arrays.asList("Apple", "Banana", "Mango", "Orange"));
List<String> preferredFruits = Arrays.asList("Apple", "Orange");

fruits.retainAll(preferredFruits);

System.out.println(fruits); // 输出:[Apple, Orange]

5. 注意事项和常见问题

注意事项

  1. 不支持 null 集合

    • 如果参数集合为 null,调用 retainAll() 会抛出 NullPointerException
      List<Integer> list = new ArrayList<>(Arrays.asList(1, 2, 3));
      list.retainAll(null); // 抛出 NullPointerException
      
  2. 原集合会被修改

    • 调用此方法后,原集合的内容会被更改,仅保留交集元素。
  3. 参数集合不可修改

    • retainAll() 不会修改参数集合。

常见问题

  • 空交集
    如果两个集合没有交集,则调用 retainAll() 后,原集合会变为空。

    List<Integer> list1 = new ArrayList<>(Arrays.asList(1, 2, 3));
    List<Integer> list2 = new ArrayList<>(Arrays.asList(4, 5, 6));
    
    list1.retainAll(list2);
    
    System.out.println(list1); // 输出:[]
    
  • 顺序保留
    如果使用 List(如 ArrayList),retainAll() 保留交集时,元素的顺序按照原集合的顺序。

    List<Integer> list1 = new ArrayList<>(Arrays.asList(1, 2, 3, 4));
    List<Integer> list2 = new ArrayList<>(Arrays.asList(4, 3));
    
    list1.retainAll(list2);
    
    System.out.println(list1); // 输出:[3, 4]
    

6. 内部原理

执行流程

  1. 遍历调用 retainAll() 的集合(如 list1)。
  2. 对于每个元素,检查它是否存在于参数集合中(如 list2)。
  3. 如果不存在,移除该元素。
  4. 返回 true,如果至少有一个元素被移除;否则返回 false

效率

  • 取决于参数集合的类型:
    • 如果参数集合是一个 HashSetretainAll() 的性能较好,因为 HashSet 提供了快速的查找操作(O(1))。
    • 如果参数集合是一个 List,性能可能会较低,因为查找操作需要线性时间(O(n))。

7. 与其他方法的区别

removeAll()

  • 区别removeAll() 是删除当前集合中与指定集合交集的元素,而 retainAll() 是保留交集元素。
  • 示例
    List<Integer> list1 = new ArrayList<>(Arrays.asList(1, 2, 3, 4));
    List<Integer> list2 = new ArrayList<>(Arrays.asList(3, 4, 5));
    
    // removeAll 删除交集
    list1.removeAll(list2);
    System.out.println(list1); // 输出:[1, 2]
    
    // retainAll 保留交集
    list1 = new ArrayList<>(Arrays.asList(1, 2, 3, 4));
    list1.retainAll(list2);
    System.out.println(list1); // 输出:[3, 4]
    

8. 总结

方法描述
作用保留集合中与指定集合交集的元素,其余元素被删除
修改原集合
参数不可为空如果参数集合为空会抛出 NullPointerException
返回值如果集合发生变化返回 true,否则返回 false
常见用途找交集、过滤集合中的元素

retainAll() 是操作集合的一个常用方法,能够帮助开发者快速进行集合间的交集操作。在使用时,需要注意原集合会被修改,因此在某些场景下可能需要备份原集合数据以避免数据丢失。


http://www.kler.cn/a/399767.html

相关文章:

  • Vue模块化开发的理解
  • 基于python Django的boss直聘数据采集与分析预测系统,爬虫可以在线采集,实时动态显示爬取数据,预测基于技能匹配的预测模型
  • 探索美赛:从准备到挑战的详细指南
  • 【c++笔试强训】(第十一篇)
  • 使用 Go 实现将任何网页转化为 PDF
  • 计算机网络:运输层 —— TCP的流量控制
  • 滑动窗口入门(LeetCode——1456定长字符串中元音字母的最大数目)
  • 【在Linux世界中追寻伟大的One Piece】手写序列化与反序列化
  • 利用hive元数据统计数据量
  • 编程之路,从0开始:结构体详解
  • 大数据技术Kafka详解 ① | 消息队列(Messages Queue)
  • 微信小程序底部button,小米手机偶现布局错误的bug
  • etcd部署(基于v3.5.15)
  • C++标准模板库 -- map和set
  • 怎么看待Ai发展前景?
  • 用python中的tkinter包实现进度条
  • 【C++】C++11 新特性揭秘:序章
  • window的Anaconda Powershell Prompt 里使用linux 命令
  • AIGC----生成对抗网络(GAN)如何推动AIGC的发展
  • 11.12.2024刷华为OD-集合的使用,递归回溯的使用
  • 【青牛科技】D2030 14W 高保真音频放大电路介绍和应用
  • 使用Mac下载MySQL修改密码
  • 【C++】了解map和set及平衡二叉树和红黑树的原理
  • 使用docker快速部署Nginx、Redis、MySQL、Tomcat以及制作镜像
  • ArcGIS Pro的arpx项目在ArcGIS Server中发布要素服务(FeatureServer)
  • ue中使用webui有效果白色拖动条 有白边