首页 > Java > java教程 > 正文

Java中Collections.emptyList和Collections.emptyMap使用

P粉602998670
发布: 2025-09-21 14:53:01
原创
987人浏览过
使用Collections.emptyList()和emptyMap()可避免null,提升代码健壮性与性能。它们返回全局唯一的不可变空集合实例,防止NPE,减少内存开销,适用于API返回值、字段默认值等无需修改的场景。

java中collections.emptylist和collections.emptymap使用

在Java开发中,

Collections.emptyList()
登录后复制
Collections.emptyMap()
登录后复制
这两个方法,说白了,就是为了提供一个不可变、类型安全且高效的空集合实例。它们的存在,很大程度上是为了帮助我们写出更健壮、更不容易出错的代码,避免那些恼人的
NullPointerException
登录后复制
,同时还能在一定程度上优化性能。在我看来,这不仅仅是几个API调用,更是一种优秀编程习惯的体现。

Collections.emptyList()
登录后复制
Collections.emptyMap()
登录后复制
都是
java.util.Collections
登录后复制
类提供的静态工厂方法。当你调用它们时,你不会得到一个新的、每次都创建的空集合对象,而是会获得一个预先定义好的、全局唯一的、不可变的空集合实例。这意味着,无论你在程序的哪个地方调用多少次,它们返回的都是同一个对象——一个没有任何元素的列表或映射,并且你无法向其中添加或删除任何元素。试图修改它们,会直接抛出
UnsupportedOperationException
登录后复制

为什么不直接返回 null,而是使用 Collections.emptyList/emptyMap?

这其实是个老生常谈的话题了,但每次提到,我总觉得还是很有必要强调一下:避免

null
登录后复制
是编写高质量代码的关键一步。我们都知道
NullPointerException
登录后复制
是Java世界里最常见的运行时错误之一,甚至被其发明者称之为“价值十亿美元的错误”。当一个方法可能返回
null
登录后复制
而调用者又忘记检查时,程序就埋下了隐患。

想象一下,你有一个服务方法,它负责查询用户列表。如果查询结果为空,你是返回

null
登录后复制
呢,还是返回一个空列表?

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

如果返回

null
登录后复制

List<User> users = userService.findUsersByCriteria(criteria);
if (users != null) { // 每次调用后都得加这个判断
    for (User user : users) {
        // ... 处理用户 ...
    }
}
登录后复制

这种代码,到处都是

if (xxx != null)
登录后复制
,看着就让人头大,而且很容易漏掉。

而如果返回

Collections.emptyList()
登录后复制

List<User> users = userService.findUsersByCriteria(criteria); // 即使没有用户,也总是一个非null的列表
for (User user : users) { // 可以直接迭代,无需担心NPE
    // ... 处理用户 ...
}
登录后复制

是不是清爽很多?调用者不需要关心底层有没有数据,只要它拿到的是一个

List
登录后复制
,就可以放心地进行迭代或其他集合操作。这极大地提升了API的健壮性和易用性。这不仅是一种编程风格,更是一种防御性编程的体现,它让你的代码在面对“无数据”这种常见情况时,能够更加优雅和稳定。

Collections.emptyList/emptyMap 与 new ArrayList()/new HashMap() 有何不同?

这两个选项虽然都能得到一个空的集合,但它们的本质和用途却大相径庭,理解它们之间的差异,能帮助你做出更明智的选择。

最核心的区别在于可变性内存效率

new ArrayList()
登录后复制
new HashMap()
登录后复制
每次调用都会在堆上创建一个全新的、可变的集合对象。这意味着你可以自由地向其中添加、删除元素。即使你创建时它是空的,它也为未来的修改做好了准备。每次创建都需要分配内存,并且可能涉及对象的初始化开销。

百度文心百中
百度文心百中

百度大模型语义搜索体验中心

百度文心百中 22
查看详情 百度文心百中

Collections.emptyList()
登录后复制
Collections.emptyMap()
登录后复制
则返回的是一个不可变单例对象。

  • 不可变性: 一旦你获得了这个空集合,你就不能向它添加任何东西。任何尝试修改它的操作都会立即抛出
    UnsupportedOperationException
    登录后复制
    。这在多线程环境下尤其重要,因为不可变对象天生就是线程安全的,无需额外的同步措施。
  • 单例: 无论你在程序的多少个地方调用它们,它们始终返回的是同一个
    EmptyList
    登录后复制
    EmptyMap
    登录后复制
    实例。这带来了显著的内存优势,特别是当你的应用中有很多地方需要表示“空”状态时,你不需要为每个“空”都创建一个新对象,从而节省了大量的堆内存,也减少了垃圾回收的压力。

所以,如果你需要一个可以随时填充数据的集合,即使它开始是空的,也应该使用

new ArrayList()
登录后复制
new HashMap()
登录后复制
。但如果你只是想表示一个“无数据”的最终状态,并且不希望它被修改,那么
Collections.emptyList()
登录后复制
Collections.emptyMap()
登录后复制
才是更优雅、更高效的选择。

在实际项目中,何时是使用 Collections.emptyList/emptyMap 的最佳时机?

在我日常的开发工作中,我发现有几个场景是

Collections.emptyList()
登录后复制
Collections.emptyMap()
登录后复制
发光发热的地方:

  • API 方法的返回值: 这是最常见的场景。当你的业务逻辑在某些条件下没有数据可返回时,比如一个查询方法没有找到匹配项,或者一个聚合操作结果为空,返回一个空集合而非

    null
    登录后复制
    是最佳实践。

    public List<Order> getRecentOrders(String userId) {
        // 假设这里是数据库查询逻辑
        List<Order> orders = databaseService.findOrdersByUserId(userId);
        if (orders.isEmpty()) {
            return Collections.emptyList(); // 明确表示没有订单,而不是null
        }
        return orders;
    }
    登录后复制
  • 类的字段默认值: 有时候一个类的某个列表或映射字段,在对象刚创建时可能没有数据,但你又不想把它初始化为

    null
    登录后复制
    ,这时
    Collections.emptyList()
    登录后复制
    Collections.emptyMap()
    登录后复制
    是一个很好的选择。

    public class Product {
        private String name;
        private List<String> tags = Collections.emptyList(); // 默认一个空标签列表,避免NPE
    
        public Product(String name) {
            this.name = name;
        }
    
        // 如果需要添加标签,通常会创建一个新的可变列表
        public void addTag(String tag) {
            if (this.tags == Collections.emptyList()) { // 如果是默认的空列表
                this.tags = new ArrayList<>(); // 第一次添加时才实例化可变列表
            }
            ((ArrayList<String>) this.tags).add(tag);
        }
    
        public List<String> getTags() {
            return tags;
        }
    }
    登录后复制

    请注意,这里

    addTag
    登录后复制
    方法的实现,如果
    tags
    登录后复制
    初始是
    Collections.emptyList()
    登录后复制
    ,需要先创建一个新的
    ArrayList
    登录后复制
    。这是一种常见的模式,即“懒初始化”或“写时复制”。

  • 作为 Stream API 的初始值或聚合结果: 在使用Java Stream API进行数据处理时,如果某个

    collect
    登录后复制
    操作可能导致空结果,返回
    Collections.emptyList()
    登录后复制
    也是一个自然的选择。

    List<String> validNames = names.stream()
                                   .filter(name -> name != null && !name.trim().isEmpty())
                                   .collect(Collectors.toList()); // collect默认不会返回null
    // 但如果你的自定义收集器或者其他逻辑可能导致空,可以考虑
    // return validNames.isEmpty() ? Collections.emptyList() : validNames;
    登录后复制

    当然,

    Collectors.toList()
    登录后复制
    本身在结果为空时就会返回一个空
    ArrayList
    登录后复制
    ,所以这个场景更多是针对一些更复杂的自定义收集器或者在
    Optional
    登录后复制
    场景下使用。

  • 在某些条件逻辑分支中: 当程序根据特定条件,需要返回一个明确的“无数据”状态时,它们同样适用。

总而言之,只要你希望表达一个集合是空的,并且它在后续的生命周期中不应该被修改,那么

Collections.emptyList()
登录后复制
Collections.emptyMap()
登录后复制
几乎总是比
null
登录后复制
new ArrayList()
登录后复制
更好的选择。它们让你的代码更清晰、更安全,也更符合Java的集合设计哲学。

以上就是Java中Collections.emptyList和Collections.emptyMap使用的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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