BigDecimal 的 Null 安全 Comparator 实现

bigdecimal 的 null 安全 comparator 实现

在 Java 中,对包含 BigDecimal 类型的列表进行排序时,如果列表中存在 null 值,直接使用 Comparator.comparing 可能会导致 NullPointerException。 错误信息显示,即使使用了 Comparator.nullsFirst,仍然会出现空指针异常,这是因为 nullsFirst 作用于 ProductBenefitResponse 对象,而不是 BigDecimal 值本身。

为了解决这个问题,我们需要创建一个专门针对 BigDecimal 类型的 null 安全 Comparator,并将其应用于 thenComparing 方法中。

实现 Null 安全的 BigDecimal Comparator

首先,我们需要创建一个 Comparator 对象,该对象能够处理 null 值并进行自然排序。可以使用 Comparator.nullsFirst 和 Comparator.naturalOrder 组合来实现:

Comparator bigDecimalComparator = Comparator.nullsFirst(Comparator.naturalOrder());

这段代码创建了一个 bigDecimalComparator,它会将 null 值排在最前面,并使用 BigDecimal 的自然顺序进行排序。Comparator.naturalOrder() 实际上就是 BigDecimal::compareTo 的简写。

应用 Null 安全 Comparator 进行排序

接下来,将这个 bigDecimalComparator 应用到列表的排序操作中。假设我们有一个 ProductBenefitResponse 对象的列表 list,并且 ProductBenefitResponse 类有一个 getDescription() 方法返回 String 类型,以及一个 getBenefitLimit() 方法返回 BigDecimal 类型,我们可以这样排序:

List list = new ArrayList(benefitMap.values());list.sort(    Comparator        .comparing(ProductBenefitResponse::getDescription)        .thenComparing(ProductBenefitResponse::getBenefitLimit, bigDecimalComparator));

在这里,thenComparing 方法接受一个额外的参数,即我们创建的 bigDecimalComparator。这样,在比较 benefitLimit 属性时,就会使用这个 null 安全的比较器,从而避免 NullPointerException。

完整示例代码

import java.math.BigDecimal;import java.util.ArrayList;import java.util.Comparator;import java.util.List;import java.util.Map;public class BigDecimalComparatorExample {    public static void main(String[] args) {        // 假设 benefitMap 是一个 Map        Map benefitMap = Map.of(            "key1", new ProductBenefitResponse("A", new BigDecimal("100.00")),            "key2", new ProductBenefitResponse("B", null),            "key3", new ProductBenefitResponse("A", new BigDecimal("50.00")),            "key4", new ProductBenefitResponse("C", new BigDecimal("200.00"))        );        List list = new ArrayList(benefitMap.values());        // 创建 BigDecimal 的 null 安全 Comparator        Comparator bigDecimalComparator = Comparator.nullsFirst(Comparator.naturalOrder());        // 使用 null 安全 Comparator 进行排序        list.sort(            Comparator                .comparing(ProductBenefitResponse::getDescription)                .thenComparing(ProductBenefitResponse::getBenefitLimit, bigDecimalComparator)        );        // 打印排序后的结果        list.forEach(productBenefitResponse -> System.out.println(productBenefitResponse.getDescription() + ": " + productBenefitResponse.getBenefitLimit()));    }    static class ProductBenefitResponse {        private String description;        private BigDecimal benefitLimit;        public ProductBenefitResponse(String description, BigDecimal benefitLimit) {            this.description = description;            this.benefitLimit = benefitLimit;        }        public String getDescription() {            return description;        }        public BigDecimal getBenefitLimit() {            return benefitLimit;        }        @Override        public String toString() {            return "ProductBenefitResponse{" +                    "description='" + description + ''' +                    ", benefitLimit=" + benefitLimit +                    '}';        }    }}

注意事项

确保 ProductBenefitResponse::getBenefitLimit 返回的是 BigDecimal 类型,而不是 BigDecimal 的包装类。如果需要将 null 值排在最后,可以使用 Comparator.nullsLast(Comparator.naturalOrder())。如果需要自定义 BigDecimal 的比较规则,可以实现自己的 Comparator。

总结

通过创建一个专门针对 BigDecimal 类型的 null 安全 Comparator,并将其应用于列表的排序操作中,可以有效地避免 NullPointerException,并确保排序结果的正确性。这种方法不仅适用于 BigDecimal,也可以推广到其他可能包含 null 值的类型的排序场景中。

以上就是BigDecimal 的 Null 安全 Comparator 实现的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/107665.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月22日 16:07:26
下一篇 2025年11月22日 16:48:10

相关推荐

  • Golang错误处理基础与常用方法

    Golang错误处理通过显式返回error值,强制开发者主动检查和处理错误,提升了代码健壮性与可预测性。 Golang的错误处理,核心在于其显式、强制的机制,它要求开发者必须主动地检查并处理每一个可能发生的错误,而非依赖隐式的异常捕获。这套机制围绕着一个简单的 error 接口展开,通过函数返回值的…

    2025年12月15日
    000
  • Go语言中可变参数的正确转发与封装实践

    本文深入探讨Go语言中可变参数(variadic functions)的正确使用与转发机制,特别是在封装如fmt.Sprintf这类函数时,如何避免将整个参数切片作为单一参数传递的常见错误。通过示例代码和详细解释,我们将学习如何使用…操作符将切片元素“展开”为独立的参数,从而确保可变参数…

    2025年12月15日
    000
  • Golang反射访问私有方法与字段实例

    在Go语言中,反射可动态获取类型信息并操作对象,但需遵循包级访问控制。2. 私有字段和方法仅在同包内可通过反射访问,跨包会触发权限限制或panic。3. 示例中通过reflect.ValueOf(&p).Elem()获取结构体字段并修改私有字段值。4. 调用私有方法同样需在同包内使用反射方法…

    2025年12月15日
    000
  • Go 语言可变参数转发:理解 … 语法

    本文探讨 Go 语言中可变参数(variadic functions)的正确传递与转发机制。当将一个可变参数列表传递给另一个可变参数函数时,常见的错误是将整个参数切片直接传递,导致意外输出。文章将详细解释为何需要使用 … 语法来“解包”切片,从而确保参数被正确地作为独立个体进行转发,避免…

    2025年12月15日
    000
  • Go语言中变长参数的正确传递姿势:深入理解 … 语法

    本文深入探讨Go语言中变长参数(variadic functions)的正确使用方法,特别是如何将一个变长参数列表无缝地传递给另一个接受变长参数的函数。我们将解释直接传递切片为何会导致 fmt 函数出现 EXTRA 错误,并详细介绍如何利用 … 语法将切片“展开”为独立的参数,从而实现参…

    2025年12月15日
    000
  • Go语言字符串切片与优雅处理尾部字符

    本文旨在澄清Go语言中字符串切片与C语言字符串处理的常见混淆,特别是关于空终止符的误解。我们将探讨Go字符串的内部机制,演示如何以惯用的方式高效移除字符串末尾的特定字符(如换行符),并通过示例代码和注意事项,帮助开发者避免不必要的复杂操作,掌握Go语言中字符串操作的正确实践。 理解Go语言字符串与切…

    2025年12月15日
    000
  • Golangpanic recover在服务守护中的应用

    答案:Go中的panic recover机制用于捕获运行时panic,防止程序崩溃。在goroutine入口通过defer+recover捕获异常,可记录日志并重启worker,提升服务健壮性;但需避免滥用,应优先使用error返回处理常规错误,仅在不可恢复场景使用panic,以减少性能开销。 pa…

    2025年12月15日
    000
  • Golang指针与结构体组合使用优化技巧

    使用指针指向结构体可避免复制开销,提升性能。在传递大型结构体时,传指针仅传递地址,减少内存占用和复制时间。如User和Image结构体示例所示,值传递会复制整个结构体,导致性能下降,而指针传递高效且能修改原数据。此外,处理嵌套指针时需检查nil,防止空指针异常,如Employee结构体中先判空emp…

    2025年12月15日
    000
  • Golang迭代器模式自定义集合遍历实践

    答案:Go中迭代器模式通过接口和结构体实现,为自定义集合提供统一遍历方式,支持状态管理、泛型增强类型安全,并适用于复杂数据结构如二叉树的遍历,相比Channel更适用于同步、单线程场景下的封装与控制。 在Golang中,当我们面对自定义的复杂数据结构时,如果想提供一种统一且不暴露内部细节的遍历方式,…

    2025年12月15日
    000
  • Golang模板方法模式与业务逻辑分离

    模板方法模式通过固定算法骨架实现业务逻辑分离,Go中用接口定义Read、Validate、Transform、Save步骤,由CSVProcessor和JSONProcessor等具体类型实现差异化处理,统一流程控制在ProcessDataTemplate函数中。 Golang中的模板方法模式提供了…

    2025年12月15日
    000
  • Golang中如何利用goroutine和channel实现非阻塞操作

    答案:通过goroutine执行任务、channel传递结果并结合select与context实现超时控制和取消信号,使主程序非阻塞。主goroutine可并发启动耗时任务,利用带缓冲channel或select的default/case实现异步通信,避免阻塞;context用于传递取消指令,防止g…

    2025年12月15日
    000
  • Go语言字符串切片:理解与惯用处理末尾字符的方法

    本教程深入探讨Go语言中字符串切片(slice)的机制,纠正关于字符串终止符和长度计算的常见误解。我们将学习如何使用Go的惯用方法高效且安全地处理字符串末尾字符,特别是移除bufio.ReadString读取输入时产生的换行符,避免C语言风格的错误操作,掌握Go字符串处理的精髓。 Go语言字符串与切…

    2025年12月15日
    000
  • Golang并发编程错误调试与日志分析

    答案:Go并发调试需结合竞态检测、结构化日志、pprof与trace工具及压力测试,系统性排查竞态、死锁等问题。启用-race可捕获内存冲突,结构化日志带唯一标识便于追踪,pprof分析goroutine阻塞,trace可视化调度时序,多核测试和Gosched模拟极端场景,预防线上故障。 Go语言的…

    2025年12月15日
    000
  • Golang服务间认证与授权实践示例

    使用JWT实现Golang微服务间认证与授权,通过HTTP中间件验证令牌并控制权限。1. 发送方生成含iss、aud声明的JWT;2. 接收方中间件校验签名、过期时间及请求头格式;3. 校验aud、iss匹配目标服务;4. 可扩展基于角色或服务名的授权逻辑;5. 结合HTTPS、密钥管理与日志审计提…

    2025年12月15日
    000
  • Golang多模块项目环境初始化实践

    多模块结构通过合理划分职责提升项目可维护性,需设计清晰目录并使用go mod init初始化各模块,通过replace解决本地依赖问题。 在现代Golang项目开发中,随着项目规模扩大,单模块管理逐渐难以满足需求。多模块(multi-module)结构能更好划分职责、提升复用性与团队协作效率。合理初…

    2025年12月15日
    000
  • Golang包路径与项目结构优化方法

    Go项目应以域名开头的全小写路径命名包,避免冲突并遵循社区规范;使用go mod init初始化模块,合理设计目录结构如cmd、internal、pkg等分层,确保包名简洁明确且与目录一致,通过Go Modules管理依赖,提升可维护性与团队协作效率。 Go语言的包路径和项目结构直接影响代码的可维护…

    2025年12月15日
    000
  • Golang循环嵌套与标签跳转使用示例

    Golang中使用标签跳转(break/continue Label)可直接跳出多层循环或跳过外层迭代,适用于需立即终止嵌套循环的场景,如搜索目标后终止;相比传统标志位更简洁,但应慎用以避免降低可读性。通过函数封装和return通常更符合Go风格,标签跳转宜在必要时用于保持代码局部性,且需确保标签命…

    2025年12月15日
    000
  • Go语言中可变参数函数的正确传递技巧

    本文旨在深入探讨Go语言中可变参数(Variadic Functions)的正确使用方法,特别是当需要将一个可变参数列表传递给另一个同样接受可变参数的函数时所遇到的常见问题及解决方案。通过分析错误示例,我们将理解为何直接传递[]interface{}类型会导致非预期输出,并介绍如何利用Go语言的&#…

    2025年12月15日
    000
  • Golang使用TLS实现安全网络通信

    Golang使用crypto/tls包实现安全通信,通过生成自签名证书(如openssl命令)用于测试,配置tls.Config加载证书并监听TLS连接;客户端需正确配置tls.Config并避免InsecureSkipVerify生产使用;可通过设置MinVersion/MaxVersion强制协…

    2025年12月15日
    000
  • Go语言SAML库实现单点登录(SSO)指南

    本文旨在解决Go语言在SAML库支持上的早期空白,为开发者提供实现单点登录(SSO)的实用指南。我们将探讨目前可用的Go语言SAML库,分析其优势,并提供集成SAML SSO的专业建议和最佳实践,帮助Go开发者高效构建安全的身份验证系统。 1. Go语言SAML库的演进与现状 在go语言发展的早期,…

    2025年12月15日
    000

发表回复

登录后才能评论
关注微信