在Java中,从List A中找出List B没有的数据(即求差集)

方法 核心思路 优点 缺点 适用场景
循环遍历 遍历A,检查元素是否不在B中 逻辑简单直观 性能较差(O(n*m)) 小数据量,注重代码可读性
Stream API 使用Stream的filter进行声明式处理 代码简洁,函数式风格 包含检查性能依赖底层集合 常用推荐,代码简洁性要求高
HashSet优化 将B转为HashSet,利用O(1)查找 性能最佳(约O(n)) 有转换开销,不保持顺序 大数据量,对性能要求高
**removeAll**​ 直接修改A,移除B中也有的元素 代码极简(一行) 直接修改原集合A 确定可修改原集合,且不需保留原顺序

下面我们通过具体代码示例来详细了解每种方法。

循环遍历

这是最基础的方法,通过循环遍历列表A,并判断每个元素是否不在列表B中。

复制代码
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

public class ListDifference {
    public static void main(String[] args) {
        List<String> a = new ArrayList<>(Arrays.asList("Apple", "Banana", "Cherry", "Date"));
        List<String> b = new ArrayList<>(Arrays.asList("Banana", "Date", "Fig"));
        
        List<String> result = new ArrayList<>();
        
        for (String item : a) {
            if (!b.contains(item)) {
                result.add(item);
            }
        }
        
        System.out.println("A有B无的元素(循环遍历): " + result); // 输出: [Apple, Cherry]
    }
}

Stream API

利用Java 8的Stream API,可以写出更简洁、更函数式的代码。

复制代码
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;

public class ListDifference {
    public static void main(String[] args) {
        List<String> a = new ArrayList<>(Arrays.asList("Apple", "Banana", "Cherry", "Date"));
        List<String> b = new ArrayList<>(Arrays.asList("Banana", "Date", "Fig"));
        
        List<String> result = a.stream()
                               .filter(item -> !b.contains(item))
                               .collect(Collectors.toList());
        
        System.out.println("A有B无的元素(Stream): " + result); // 输出: [Apple, Cherry]
    }
}

HashSet优化

当数据量较大时,先将列表B转换为HashSet可以大幅提升查找效率。

复制代码
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;

public class ListDifference {
    public static void main(String[] args) {
        List<String> a = new ArrayList<>(Arrays.asList("Apple", "Banana", "Cherry", "Date"));
        List<String> b = new ArrayList<>(Arrays.asList("Banana", "Date", "Fig"));
        
        // 将List B转换为HashSet
        Set<String> setB = new HashSet<>(b);
        
        List<String> result = a.stream()
                               .filter(item -> !setB.contains(item))
                               .collect(Collectors.toList());
        
        System.out.println("A有B无的元素(HashSet优化): " + result); // 输出: [Apple, Cherry]
    }
}

使用 removeAll

ListremoveAll方法会直接修改原列表,使用前需谨慎。

复制代码
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

public class ListDifference {
    public static void main(String[] args) {
        List<String> a = new ArrayList<>(Arrays.asList("Apple", "Banana", "Cherry", "Date"));
        List<String> b = new ArrayList<>(Arrays.asList("Banana", "Date", "Fig"));
        
        // 创建a的副本,避免修改原列表
        List<String> aCopy = new ArrayList<>(a);
        aCopy.removeAll(b);
        
        System.out.println("原列表A未被修改: " + a); // 输出: [Apple, Banana, Cherry, Date]
        System.out.println("A有B无的元素(removeAll): " + aCopy); // 输出: [Apple, Cherry]
        
        // 如果确定要修改原列表A,可以直接使用:a.removeAll(b);
        // a.removeAll(b);
        // System.out.println("修改后的列表A: " + a); // 输出: [Apple, Cherry]
    }
}
相关推荐
金銀銅鐵10 小时前
[Python] 基于欧几里得算法,实现分数约分计算器
python·数学
Lyn_Li12 小时前
Kaggle Top 5 | 198只股票、200条数据的金融预测——BattleFin高分方案从零复现
python·kaggle·比赛复盘·金融预测
小九九的爸爸17 小时前
前端想要入门Agent开发,要具备哪些Python基础?
python·agent·ai编程
阿耶同学18 小时前
手把手教你用 LangGraph 搭建三层嵌套 Agent 架构
python·程序员
花酒锄作田1 天前
Pydantic校验配置文件
python
hboot1 天前
AI工程师第四课 - 深度学习入门
pytorch·python·神经网络
ZhengEnCi2 天前
P2M-Matplotlib折线图完全指南-从数据可视化到趋势分析的Python绘图利器
python·matlab·数据可视化
ZhengEnCi2 天前
P2L-Matplotlib饼图完全指南-从数据可视化到图表定制的Python绘图利器
python·matlab
曲幽2 天前
你的REST接口还在“过度投喂”数据吗?——FastAPI + GraphQL实战避坑指南
python·fastapi·web·graphql·route·cors·rest·strawberry
用户8358086187912 天前
基于 Self-RAG 与列表级重排序的进阶 RAG 系统设计与实现
python