ArrayList()和Collections.emptyList()的区别emptyList()、emptySet()、emptyMap()的作用和好处以及要注意的地方

前言

Java中ArrayList或许是我们平时开发最常用的一个集合类了,其次是HashMap,基本上满足了业务开发的绝大多数场景。今天要说的就是Collections.emptyList()和new ArrayList()的区别以及注意事项。

先来一段代码

10a9f16b9123bd5bec7e814df6a05f6f.png

运行main方法,会有如下输出:

d47ef48469f852b9a5f84c61c2d187eb.png

很显然,Collections.emptyList()会抛出“java.lang.UnsupportedOperationException”的异常。

使用及区别

日常开发中,我们经常会写一个方法,返回一个集合。当这个方法返回的数据为空时候,通常我们会返回一个Collections.emptyList(),而不是null。这样方法调用者就不用担心集合是否null了。比如这样的:

67b2e74a24ad67435dd5a1d885785c10.png

突然有一天,有一个同事调用了你的这个方法,然后再加入自己的数据:

debab80b65d1f881f4e5fb5fdc6ad6cd.png

就会可能出现如下几种情况:

  1. 自测时候被发现:幸亏本大神自测发现,不然测试那帮家伙肯定发现不了;
  2. 被测试发现:那是因为没有数据,好了好了,我兼容一下好了;
  3. 代码上线了:不断的抛出异常,不断的报警,大多数用户访问的页面出现系统异常。

很不幸,我们真的成功的走到了第三步,然后被用户反应出来了(因为只有部分用户在没有数据的情况下会产生这个bug)。。。


说了这么多,那既然Collections.emptyList()有问题我就直接用new ArrayList()没问题了吧?

是的!没有问题,但是总感觉low了点?

所以,搞清楚二者的区别以及适用场景才是一个爱学习的程序员要做的事情。

先看Collections.emptyList()方法的源码

ad6acc877c80f4bbe1587a05947f3149.png

返回一个不可变的空集合,那如何是不可变的呢?

f9289d68eade503dc9a14191dbff8060.png

原来是EmptyList类没有实现add()和remove()方法。

那使用这个方法的意义是什么?或者说JDK为什么要提供这个方法呢?

大家肯定看到了EMPTY_LIST这个类常量!

2e128068010ec204dc79fd94ed2eb6f2.png

它在Collections类里面属于静态常量,静态常量什么概念?在JVM虚拟机加载完毕时候就已经存在了,当我们调用这个方法的时候不需要再去创建一个新的List对象了,减少了内存开销。所以当你的方法调用频率很高,并且可能会返回空集合时候,使用Collections.emptyList()会提高你的代码性能,降低内存开销。

总结

  1. Collections.emptyList()返回了一个不可变的空集合,不支持集合数据修改操作,多次调用不会额外增加内存消耗;
  2. new ArrayList()每次都会创建一个对象,需要内存开销;
  3. Collections.emptyList()使得调用者不需要去判断返回是否为null,但是需要注意这个集合不可变;
  4. EMPTY_LIST和emptyList()的唯一区别就是EMPTY_LIST没有支持泛型,需要强转一下;
  5. emptySet()、emptyMap()和emptyList()是一个道理;
  6. Java 9中的List.of()简化了emptyList()的使用。



背景

项目中有时候会使用Collections.emptyList返回一个空列表,但是emptyList在执行add,remove等方法时会直接抛出UnsupportedOperationException异常,我们可以看下源码

public class Collections {public static final List EMPTY_LIST = new EmptyList<>();public static final <T> List<T> emptyList() {return (List<T>) EMPTY_LIST;}private static class EmptyList<E>extends AbstractList<E>implements RandomAccess, Serializable {public int size() {return 0;}public boolean isEmpty() {return true;}public boolean contains(Object obj) {return false;}public E get(int index) {throw new IndexOutOfBoundsException("Index: "+index);}}
}public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E> {public boolean add(E e) {add(size(), e);return true;}public void add(int index, E element) {throw new UnsupportedOperationException();}
}

我们可以发现emptyList最后执行的是AbstractList里面的add方法,所以会直接抛出异常。为了避免报错,有同事提议将emptyList都用new ArrayList()代替,此时决定看下emptyList的优势

代码

通过百度知道emptyList不需要占用内存,而ArrayList每次new都会在堆中开辟内存空间存放对象,我们先通过代码验证一下

public class ListTest {private static final int printCount = 10000;public static void main(String[] args) {long freeMemory = Runtime.getRuntime().freeMemory();System.out.println("freeMemory: " + freeMemory);for (int i = 0; i < printCount; i++) {List newList = new ArrayList();}long freeMemoryNew=Runtime.getRuntime().freeMemory();System.out.println("freeMemory use: "+(freeMemory-freeMemoryNew));for(int i = 0;i < printCount; i++){List emptyList = Collections.emptyList();}long freeMemoryEmpty = Runtime.getRuntime().freeMemory();System.out.println("freeMemory use: "+(freeMemoryNew-freeMemoryEmpty));}
}

此时我们看一下执行结果

Connected to the target VM, address: '127.0.0.1:63534', transport: 'socket'
Disconnected from the target VM, address: '127.0.0.1:63534', transport: 'socket'
freeMemory: 253398816
freeMemory use: 1430376
freeMemory use: 0Process finished with exit code 0

我们可以看出new ArrayList执行一万次会消耗1430376KB内存,而Collections.emptyList不会消耗内存,那有人会说emptyList不也是new EmptyList()吗?其实我们再仔细看下上面的源码就发现emptyList是一个static变量,只会初始化一次,所以后续使用不会再初始化对象。此时我们可以得出结论,emptyList不占用内存,但是无法执行add等方法,new ArrayList()占用内存,但是会初始化对象数组,可以执行add等方法。




Java之Collections.emptyList()、emptySet()、emptyMap()的作用和好处以及要注意的地方

先说明一下好处有哪些:
1,如果你想 new 一个空的 List ,而这个 List 以后也不会再添加元素,那么就用 Collections.emptyList() 好了。
new ArrayList() 或者 new LinkedList() 在创建的时候有会有初始大小,多少会占用一内存。
每次使用都new 一个空的list集合,浪费就积少成多,浪费就严重啦,就不好啦
2,为了编码的方便。
比如说一个方法返回类型是List,当没有任何结果的时候,返回null,有结果的时候,返回list集合列表。
那样的话,调用这个方法的地方,就需要进行null判断。使用emptyList这样的方法,可以方便方法调用者。返回的就不会是null,省去重复代码。

注意的地方:
这个空的集合是不能调用.add(),添加元素的。因为直接报异常。因为源码就是这么写的:直接抛异常。

哦,Collections里面没这么写,但是EmptyList继承了AbstractList这个抽象类,里面简单实现了部分集合框架的方法。
这里面的add方法最后调用的方法体,就是直接抛异常。
throw new UnsupportedOperationException();
这么解释add报异常就对啦。

下面简单看下这个源码:

   /** * Collections 类里面的方法如下,一步步往下看就是啦 */  public static final <T> List<T> emptyList() {  return (List<T>) EMPTY_LIST;  }  
//。。。。。  /** * Collections 类里面的方法如下,一步步往下看就是啦 */  public static final List EMPTY_LIST = new EmptyList<>();  
//。。。。。  /** * Collections里面的一个静态内部类 */  private static class EmptyList<E> extends AbstractList<E> implements RandomAccess, Serializable {  private static final long serialVersionUID = 8842843931221139166L;  public Iterator<E> iterator() {  return emptyIterator();  }  public ListIterator<E> listIterator() {  return emptyListIterator();  }  public int size() {return 0;}  public boolean isEmpty() {return true;}  public boolean contains(Object obj) {return false;}  public boolean containsAll(Collection<?> c) { return c.isEmpty(); }  public Object[] toArray() { return new Object[0]; }  public <T> T[] toArray(T[] a) {  if (a.length > 0)  a[0] = null;  return a;  }  public E get(int index) {  throw new IndexOutOfBoundsException("Index: "+index);  }  public boolean equals(Object o) {  return (o instanceof List) && ((List<?>)o).isEmpty();  }  public int hashCode() { return 1; }  // Preserves singleton property  private Object readResolve() {  return EMPTY_LIST;  }  }  

除了这个emptyList,之外,还有类似的,emptyMap,emptySet等等。具体看下图,都是一个套路。

img

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/555292.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于mysql搭建框架环境搭建_Maven+Spring+Spring MVC+MyBatis+MySQL,搭建SSM框架环境

项目建设完成之后的结构&#xff1a;数据库的表结构如下&#xff1a;环境建设&#xff1a;搭建Maven环境、Tomcat环境、需要MySql 数据库支持&#xff0c;使用的编程工具Eclipse (这些是前期准备)&#xff1b;开始创建工程&#xff1a;1.创建一个Maven工程&#xff1a;选择weba…

DataIntegrityViolationException: Error attempting to get column处理方案汇总

项目背景 项目整体采用的是springbootmybatis 方式。有一次做数据查询的时候。console突然报&#xff1a;DataIntegrityViolationException: Error attempting to get column ‘xx’…异常。起初没在意。以为是xml中的SQL写错了&#xff0c;排查了没问题。百度一下这个报错&…

Mybatis原理:结果集封装详解

​ 经过sql参数解析、sql动态组装和执行sql&#xff0c;相对而言&#xff0c;结果集的封装&#xff0c;是mybatis数据处理的最后一环。这里只对查询结果而言&#xff0c;因为更新语句一般都是返回影响的行数。抛开mybatis&#xff0c;如果让我们组装结果&#xff0c;我们该如何…

python内置函数面向对象_Pyhton——面向对象进阶二:类的内置函数补充、描述符...

Pyhton——面向对象进阶二&#xff1a;一、类的内置函数补充1、isinstance(obj,cls)——检查obj是否是该类的对象class Hoo:def __init__(self,name,tem):self.name nameself.tem temclass foo(Hoo):passf1foo(e,20)print(isinstance(f1,Hoo))首先 f1 肯定是 foo 的对象&…

vue项目打包后部署到服务器(超详细步骤)

耽误了几天, 终于开始写第二篇博客了, 这篇会讲怎么将vue项目打包部署到服务器, 其实和上一篇的uni-app步骤一样的, 就是最后多了一步修改nginx配置, 好 , 上操作 一 ,打包项目 vscode下载链接&#xff1a;https://pan.baidu.com/s/1ibHt7XB6EZy37BDb1CigWw 提取码&#xff1…

postman怎么不登陆使用_最新百度云不限速,免安装、免登陆、不限速,打开网站就能使用...

上次给大家安利了一波Pandownload手机版/电脑版。那篇文章中也说了&#xff0c;这类应用使用不当可能会遇到账号被限速的情况&#xff0c;而且手机版必须登录才能进行不限速下载。总之&#xff0c;凡是没登录账号的小伙伴&#xff0c;下载过程会非常曲折。那么是否有无需登录就…

vue项目配置打包测试环境/生产环境

vue项目配置打包测试环境/生产环境&#xff1a; 开发环境运行命令&#xff1a;npm run serve 生产环境打包命令&#xff1a;npm run pro 测试环境打包命令&#xff1a;npm run build 步骤&#xff1a; 1.项目中添加一个配置ip的js文件&#xff0c;比如如下的ip-config.js&…

SpringMVC访问WEB-INF下的jsp解决方案Spring Boot集成使用jsp

SpringMVC访问WEB-INF下的jsp解决方案 一. 问题 ​将项目中用到的jsp等文件放在WEB-INF目录下。实际开发过程中&#xff0c;需要在框架页面通过iframe嵌入对应的具体页面&#xff0c;此处如果直接调用对应页面所在的url地址&#xff0c;则会提示404错误。 ​ WEB-INF目录下的…

SpringBoot | 详解SpringBoot配置文件及其原理

文章目录一、配置文件二、YAML语法1、基本语法2、值的写法(1)、字面量&#xff1a;普通的值&#xff08;数字&#xff0c;字符串&#xff0c;布尔&#xff09;(2)、对象、Map&#xff08;属性和值&#xff09;(3)、数组&#xff08;List、Set&#xff09;三、配置文件值注入1、…

【深入理解JVM】:Java内存模型JMM

多任务和高并发的内存交互 多任务和高并发是衡量一台计算机处理器的能力重要指标之一。一般衡量一个服务器性能的高低好坏&#xff0c;使用每秒事务处理数&#xff08;Transactions Per Second&#xff0c;TPS&#xff09;这个指标比较能说明问题&#xff0c;它代表着一秒内服…

You may use special comments to disable some warnings.Use // eslint-disable-next-line to ignore...

vue没写什么爆红 错误原因 ESLint] 对语法的要求过于严格导致编译的时候报上图那些错误。 要知道&#xff0c;这并不是代码有异常&#xff0c;而是代码格式有问题&#xff0c;这些错误并不会影响代码的执行结果。 解决方法 很简单&#xff0c;既然是ESLint 语法错误&#xff…

java jni 原理_JNI的实现原理

JNI接口函数和指针Java虚拟机访问本机代码通过调用JNI的功能特性。JNI的功能都可以通过一个接口指针。一个接口指针是一个指向指针的指针。这个指针指向一个一个指针数组&#xff0c;这个数组中的每一个成员指向一个函数入口。每个接口的功能是在一个预定义的内部数组的偏移量。…

jQuery ajax读取本地json文件_jQuery请求本地JSON文件,在谷歌浏览器运行时报跨域错误_Vscode使用Live Server

json文件 {"first":[{"name":"张三","sex":"男"},{"name":"李四","sex":"男"},{"name":"王武","sex":"男"},{"name":"李…

Vue打包并发布项目

一、 打包vue项目步骤&#xff1a; 1、对当前vue项目进行打包的命令如下&#xff1a; npm run build2、打包完成&#xff0c;会输出Build complete并且在vue项目中会生成一个名字为dist的打包文件。如下图&#xff1a; 二、 使用静态服务器工具包发布打包的vue项目 1、首先…

Grammarly:最优秀的日常英文写作辅助工具——论文英文校验

使用Grammarly也有一些心得&#xff0c;在此分享给大家。 1&#xff0c;Grammarly是什么&#xff1f; Grammarly是一款在线语法纠正和校对工具&#xff0c;支持Windows、Mac、iOS和Android等多个平台。它能够检查单词拼写、纠正标点符号、修正语法错误、调整语气以及给出风格…

SpringMVC @RequestBody和@ResponseBody原理解析

SpringMVC RequestBody和ResponseBody原理解析 前言 RequestBody作用是将http请求解析为对应的对象。例如&#xff1a; http请求的参数&#xff08;application/json格式&#xff09;&#xff1a; {"accountId": 10,"adGroupId": "12345678",…

java 高性能缓存_高性能Java缓存----Caffeine

简单介绍Caffeine是新出现的一个高性能的Java缓存&#xff0c;有了它完全可以代替Guava Cache&#xff0c;来实现更加高效的缓存&#xff1b;Caffeine采用了W-TinyLFU回收策略&#xff0c;集合了LRU和LFU的优点&#xff0c;提供了一个最佳的命中率&#xff0c;在效率上可以秒杀…

@ResponseBody 转化成json后与实体类字段名不一致_SpringMVC字符串解析成json对象(@RequestBody注解和@ResponseBody注解)

ResponseBody 转化成json后与实体类字段名不一致 实体类A字段名由B改成C后&#xff0c;Controller 中返回的List中字段名仍然是C 经过ResponseBody返回到前台后又变成了B 后来发现公司项目采用的是阿里的fastjson&#xff0c; 是开源的Json格式化工具库 此工具库是根据实体类…

java togglebutton_ToggleButton和Switch使用大全

本文转载自&#xff1a;Android零基础入门第21节&#xff1a;ToggleButton和Switch使用大全http://www.apkbus.com/blog-205190-68463.html(出处: 安卓巴士 - 安卓开发 - Android开发 - 安卓 - 移动互联网门户)&#xff0c;转载应备注出处&#xff0c;尊重原创上期学习了CheckB…

MyBatis之工作原理,简单实体的增加、修改、删除、查询_Mybatis-原理总结

一、MyBatis之工作原理 MyBatis是一个半自动映射框架。所谓半自动&#xff0c;是相对Hibernate全表映射而言的&#xff0c;MyBatis需要手动匹配提供POJO、SQL和映射关系。 我们知道&#xff0c;JDBC有四个核心对象&#xff1a; &#xff08;1&#xff09;DriverManager&#…