0

0

Java TreeMap 字符串键的自定义数值降序排序指南

霞舞

霞舞

发布时间:2025-11-27 11:17:03

|

577人浏览过

|

来源于php中文网

原创

Java TreeMap 字符串键的自定义数值降序排序指南

本文深入探讨了java `treemap`在处理字符串类型键时,如何实现非默认的自定义排序逻辑。针对`treemap`默认的字典序排序无法满足数值或长度降序排列的需求,文章详细介绍了通过提供定制化的`comparator`来转换字符串为数值进行比较,并实现降序排列的方法,确保键的排序符合预期。

理解 TreeMap 的默认排序行为

TreeMap是Java集合框架中一个基于红黑树实现的Map接口,它能保证其键(key)是按自然顺序(natural ordering)排序的,或者根据在创建TreeMap时提供的Comparator进行排序。对于String类型的键,其自然顺序是字典序(lexicographical order)。这意味着TreeMap会按照字符的Unicode值逐个比较字符串,例如 "10" 会排在 "2" 之前,因为字符 '1' 的Unicode值小于 '2'。

考虑以下示例代码:

import java.util.Map;
import java.util.TreeMap;

public class ApplicationMain {
    public static void main(String[] args) {
        final Map<String, Integer> sampleTreeMap = new TreeMap<>();
        sampleTreeMap.put("5903766131", 6);
        sampleTreeMap.put("5903767", 7);
        sampleTreeMap.put("590376614", 5);
        sampleTreeMap.put("5903766170", 9);
        sampleTreeMap.put("59037662", 12);
        sampleTreeMap.put("5903766410", 10);

        System.out.println("默认 TreeMap 排序结果:");
        sampleTreeMap.entrySet().stream().forEach(entry ->{
            System.out.println("Key : " + entry.getKey() + " -- Value : " + entry.getValue());
        });
    }
}

其输出结果将是:

默认 TreeMap 排序结果:
Key : 5903766131 -- Value : 6
Key : 590376614 -- Value : 5
Key : 5903766170 -- Value : 9
Key : 59037662 -- Value : 12
Key : 5903766410 -- Value : 10
Key : 5903767 -- Value : 7

可以看到,"5903766131" 排在 "59037662" 之前,因为在第三个字符 '0' 和 '6' 的比较中,'0' 小于 '6'。这与我们期望的将字符串作为数值进行降序排序(即数值越大、位数越多的排在前面)的需求不符。

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

实现自定义数值降序排序

当String类型的键实际上代表数值,并且需要按照这些数值的大小进行排序时,我们不能依赖String的默认字典序。解决方案是为TreeMap提供一个自定义的Comparator。这个Comparator的任务是将String键转换为数值类型(例如Long),然后根据这些数值进行比较。

使用 Comparator.comparingLong 实现

Java 8 引入的Comparator接口提供了强大的静态方法来简化比较器的创建。comparingLong方法可以接受一个函数,该函数将比较对象转换为long类型,然后基于long值进行比较。结合reversed()方法,可以轻松实现降序排列。

DreamStudio
DreamStudio

SD兄弟产品!AI 图像生成器

下载
import java.util.Comparator;
import java.util.Map;
import java.util.TreeMap;

public class ApplicationMain {
    public static void main(String[] args) {
        // 创建一个自定义 Comparator,将 String 键转换为 Long 进行比较,并按降序排列
        final Map<String, Integer> sampleTreeMap = 
            new TreeMap<>(Comparator.comparingLong((String s) -> Long.parseLong(s)).reversed());

        sampleTreeMap.put("5903766131", 6);
        sampleTreeMap.put("5903767", 7);
        sampleTreeMap.put("590376614", 5);
        sampleTreeMap.put("5903766170", 9);
        sampleTreeMap.put("59037662", 12);
        sampleTreeMap.put("5903766410", 10);

        System.out.println("
自定义数值降序排序结果:");
        sampleTreeMap.entrySet().stream().forEach(entry ->{
            System.out.println("Key : " + entry.getKey() + " -- Value : " + entry.getValue());
        });
    }
}

运行上述代码,输出将符合我们的预期:

自定义数值降序排序结果:
Key : 5903766410 -- Value : 10
Key : 5903766170 -- Value : 9
Key : 5903766131 -- Value : 6
Key : 590376614 -- Value : 5
Key : 59037662 -- Value : 12
Key : 5903767 -- Value : 7

代码解析:

  • new TreeMap<>(...): 在创建TreeMap时传入一个Comparator实例。
  • Comparator.comparingLong(...): 这是一个静态方法,它接受一个Function作为参数。这个Function的职责是将String类型的键转换为long类型。
  • (String s) -> Long.parseLong(s): 这是一个Lambda表达式,实现了Function接口。它接收一个String s,并使用Long.parseLong(s)将其解析为long值。
  • .reversed(): 在comparingLong生成的比较器基础上调用此方法,将排序顺序反转,实现降序排列。

注意事项与扩展

  1. 非数字字符串处理: 上述解决方案假设所有String键都能被成功解析为Long类型。如果TreeMap中可能包含非数字的字符串键,Long.parseLong()将会抛出NumberFormatException。在这种情况下,你需要更复杂的Comparator来处理异常或定义一个备用排序逻辑(例如,将非数字字符串排在末尾,或者按照字符串长度排序)。

    例如,一个更健壮的Comparator可能如下所示:

    new TreeMap<>( (s1, s2) -> {
        try {
            Long l1 = Long.parseLong(s1);
            Long l2 = Long.parseLong(s2);
            return l2.compareTo(l1); // 降序
        } catch (NumberFormatException e) {
            // 处理非数字字符串:例如,将非数字字符串排在数字字符串之后
            // 或者根据字符串本身进行字典序比较
            boolean isS1Numeric = s1.matches("\d+");
            boolean isS2Numeric = s2.matches("\d+");
    
            if (isS1Numeric && !isS2Numeric) return -1; // s1是数字,s2不是,s1在前
            if (!isS1Numeric && isS2Numeric) return 1;  // s1不是数字,s2是,s2在前
    
            // 都是非数字或都是数字(但前面已处理),按字符串本身比较
            return s2.compareTo(s1); // 对于非数字字符串也尝试降序
        }
    });
  2. 性能考量: 在每次比较时都进行Long.parseLong()操作可能会带来一定的性能开销,尤其是在TreeMap包含大量元素或频繁进行插入/删除操作时。对于性能敏感的场景,可以考虑:

    • 如果键的数量是固定的,可以先将键解析为Long,然后存储在TreeMap<Long, Integer>中,但根据原问题,键必须保持String类型。
    • 预处理:在将数据放入TreeMap之前,如果可能,可以先对数据进行一次排序,或者使用一个辅助的数据结构来存储解析后的Long值。
  3. 其他排序需求: 如果需求是根据字符串的长度进行降序排序,当长度相同时再按数值降序,Comparator可以这样构建:

    new TreeMap<>(Comparator
        .comparingInt((String s) -> s.length()) // 首先按长度升序
        .thenComparingLong((String s) -> Long.parseLong(s)) // 长度相同,按数值升序
        .reversed() // 最后反转整个顺序,实现长度降序,数值降序
    );

    或者更直接的实现:

    new TreeMap<>((s1, s2) -> {
        int lengthCompare = Integer.compare(s2.length(), s1.length()); // 长度降序
        if (lengthCompare != 0) {
            return lengthCompare;
        }
        return Long.compare(Long.parseLong(s2), Long.parseLong(s1)); // 数值降序
    });

总结

TreeMap是一个功能强大的有序映射,但其默认的排序行为可能不总是符合我们的特定需求。通过提供一个自定义的Comparator,我们可以完全控制键的排序逻辑。对于String类型的键,当它们代表数值时,利用Comparator.comparingLong()结合reversed()方法是实现数值降序排序的简洁高效方案。在实际应用中,务必考虑非数字字符串的处理和潜在的性能影响,并根据具体需求灵活调整Comparator的实现。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1051

2023.08.02

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1570

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

651

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1229

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1205

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

193

2025.07.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.4万人学习

Java 教程
Java 教程

共578课时 | 82.3万人学习

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

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