0

0

在Java列表中高效查找指定值之后(或最近)的元素:利用二分查找优化性能

碧海醫心

碧海醫心

发布时间:2025-07-14 21:28:38

|

401人浏览过

|

来源于php中文网

原创

在Java列表中高效查找指定值之后(或最近)的元素:利用二分查找优化性能

本文探讨了如何在Java对象列表中高效查找指定值之后或最接近的元素,尤其当列表数据量较大时。针对传统迭代遍历的低效性,文章详细介绍了如何利用Collections.binarySearch()方法结合自定义比较器实现对特定字段(如b值)的快速查找。通过示例代码,阐述了二分查找的原理、实现细节以及如何处理查找结果以定位目标元素,从而将时间复杂度从O(n)优化至O(log n)。

引言:问题背景与挑战

在软件开发中,我们经常需要处理包含大量自定义对象的数据集合。假设我们有一个row对象列表,每个row对象包含两个整型属性a和b:

class Row {
   int a;
   int b;
}

现在面临一个挑战:我们需要编写一个函数,给定一个整数x和一个Row对象列表,该函数的目标是找到列表中b值“紧随”x的Row对象。这里的“紧随”通常意味着b值大于或等于x的第一个元素,或者在x大于所有b值时,返回列表中最大的b值对应的Row。

对于小规模数据(例如几十条记录),简单的迭代遍历(O(n)时间复杂度)或许可以接受。但当数据量达到1000条甚至更多时,每次查询都遍历整个列表将导致显著的性能瓶颈。因此,寻找一种更高效的查找策略至关重要。

解决方案:基于二分查找的优化

为了解决大规模数据查找效率低下的问题,我们可以利用Java集合框架提供的Collections.binarySearch()方法。二分查找(Binary Search)是一种在有序数组中查找特定元素的算法,其时间复杂度为O(log n),相较于线性遍历的O(n)有显著的性能提升。

使用Collections.binarySearch()的前提是:目标列表必须是已排序的,并且排序规则必须与二分查找时使用的比较器(Comparator)一致。在本例中,我们需要根据Row对象的b属性进行查找,因此列表必须按b值升序排序。

立即学习Java免费学习笔记(深入)”;

实现细节与代码示例

下面我们将通过一个完整的Java代码示例来演示如何实现这一高效查找功能。

1. Row 类的定义

首先,定义我们的Row类,并为其添加构造函数、getter方法和toString方法,以便于演示和调试。

import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;
import java.util.stream.Collectors;

static class Row {
    int a, b;

    public int getA() { return a; }
    public int getB() { return b; }

    Row(int a, int b) {
        this.a = a;
        this.b = b;
    }

    @Override
    public String toString() {
        return "Row(" + a + ", " + b + ")";
    }
}

2. 自定义比较器 ORDER_BY_B

由于我们需要根据b值进行排序和查找,因此需要定义一个Comparator来指定Row对象之间如何根据b值进行比较。

XPaper Ai
XPaper Ai

AI撰写论文、开题报告生成、AI论文生成器尽在XPaper Ai论文写作辅助指导平台

下载
static final Comparator ORDER_BY_B = Comparator.comparing(Row::getB);

这个比较器ORDER_BY_B将用于对Row列表进行排序,并作为binarySearch方法的参数,指导其如何比较元素。

3. 核心查找方法 find

这是实现高效查找的关键部分。find方法接受一个整数x和Row对象列表,并返回符合条件的Row对象。

static Row find(int x, List rows) {
    int size = rows.size();
    // 1. 执行二分查找
    // Collections.binarySearch(list, key, comparator)
    // list: 必须是已按comparator排序的列表
    // key: 用于查找的虚拟对象,其b值设置为x
    // comparator: 用于比较list元素和key
    int i = Collections.binarySearch(rows, new Row(0, x), ORDER_BY_B);

    // 2. 解析binarySearch的返回值,计算实际目标索引
    int index;
    if (i >= 0) {
        // 情况1: 精确匹配
        // 如果找到了b值等于x的元素,i就是其索引。
        index = i;
    } else {
        // 情况2: 未找到精确匹配
        // 当未找到时,binarySearch返回 (-(插入点) - 1)。
        // 因此,实际的插入点(即第一个大于x的元素的索引)是 -(i + 1)。
        int insertionPoint = -(i + 1);

        if (insertionPoint >= size) {
            // 子情况2.1: x大于列表中所有元素的b值
            // 插入点在列表末尾或之后,意味着x比所有元素的b值都大。
            // 根据需求,此时我们返回列表中b值最大的元素(即最后一个元素)。
            index = size - 1;
        } else {
            // 子情况2.2: x小于或介于列表中某些元素的b值之间
            // insertionPoint就是第一个b值大于或等于x的元素的索引。
            index = insertionPoint;
        }
    }

    // 确保索引在有效范围内,防止空列表或极端情况下的越界
    // (虽然上述逻辑已大部分覆盖,但作为通用健壮性考虑)
    if (size == 0) return null; // 列表为空
    if (index < 0) index = 0; // x小于所有元素,取第一个
    if (index >= size) index = size - 1; // x大于所有元素,取最后一个 (此行通常被 insertionPoint >= size 逻辑处理)

    return rows.get(index);
}

binarySearch返回值解析:

  • i >= 0: 表示在列表中找到了一个与查找键(这里是b值为x的Row)精确匹配的元素,i就是该元素的索引。
  • i 表示列表中没有找到精确匹配的元素。此时,返回值为一个负数,其绝对值减1(即-(i + 1))表示如果将查找键插入到列表中以保持其排序顺序,它将被插入的索引位置。
    • 例如,如果binarySearch返回-1,则-(i + 1)为0,表示x比列表中所有元素都小,它应该插入到索引0的位置。
    • 如果binarySearch返回-(size + 1),则-(i + 1)为size,表示x比列表中所有元素都大,它应该插入到列表末尾。

基于这种特性,我们的index计算逻辑能够准确地定位到“第一个b值大于或等于x的元素”,或者在x大于所有b值时,定位到列表中b值最大的元素。

4. 测试与演示 main 方法

最后,通过一个main方法来测试我们的find函数。注意,在调用find之前,我们必须确保rows列表是按照ORDER_BY_B比较器排序的。

public static void main(String[] args) {
    List rows = Arrays.asList(
        new Row(20, 2),
        new Row(40, 4),
        new Row(50, 5),
        new Row(70, 7));

    // 关键步骤:确保列表按b值排序,这是二分查找的前提
    // 如果原始列表rows已经根据a值排序且a的排序隐含了b的排序,
    // 则可能不需要再次排序。但为确保通用性和正确性,显式排序是最佳实践。
    List orderByB = rows.stream().sorted(ORDER_BY_B).collect(Collectors.toList());

    System.out.println("Sorted List: " + orderByB); // 打印排序后的列表

    for (int i = 0; i < 9; ++i) {
        System.out.println("查找 " + i + " : " + find(i, orderByB));
    }
}

运行结果示例:

Sorted List: [Row(2

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

825

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

724

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

728

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

395

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

445

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

428

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16881

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号