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 5.6加用户_Mysql 5.6添加修改用户名和密码的方法

先登录MySQLshell> mysql --userroot mysql有密码的需要添加 –password 或-p 选项添加用户mysql>CREATE USER finleylocalhost IDENTIFIED BY some_pass;mysql>GRANT ALL PRIVILEGES ON *.* TO finleylocalhost WITH GRANT OPTION;mysql>CREATE USER finley% IDEN…

java中SimpleDateFormat线程安全问题及解决方案

最近看到一篇文章提到了SimpleDateFormat这个类。说这个类在单线程程序中没问题&#xff0c;但是在多线程环境下会线程安全的问题。 出于兴趣对这个问题进行了查证。网上有很多关于这个问题的文章&#xff0c;也解析了其中的原因。主要原因是因为SimpleDateFormat转换日期是通…

基于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;排查了没问题。百度一下这个报错&…

mysql 查询商品列表 显示tag_让前台页面商品列表显示后台数据库中的商品

品牌&#xff1a;经销商&#xff1a;规格&#xff1a;每零售价&#xff1a;元购买日期&#xff1a;件&#xffe5;19.6&#xffe5;19.6删除实现单选&#xff0c;全选商品function checkAll(obj, name){var el document.getElementsByTagName(input);var len el.length;for (…

servlet的本质是什么,它是如何工作的?

servlet的本质是什么&#xff0c;它是如何工作的&#xff1f;基于计算机和网络通讯(主要是http协议)构建网络应用的&#xff0c;具体的实现细节是怎样的? 查看https://www.zhihu.com/question/21416727

oschina mysql limit_如何提高MySQL Limit查询的性能

在MySQL数据库操作中&#xff0c;我们在做一些查询的时候总希望能避免数据库引擎做全表扫描&#xff0c;因为全表扫描时间长&#xff0c;而且其中大部分扫描对客户端而言是没有意义的。其实我们可以使用Limit关键字来避免全表扫描的情况&#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&…

mysql命令实践_MySQL:常用命令行

登入mysql -h192.168.1.110 -uroot -ppassword登出quit/exit查看数据库show databases;用户权限#添加grant select on db.table to userhost;grant select,update on *.* to test%;#撤销revoke all on *.* from test%;#查看show grants;show grants for userlocalhost#删除用户…

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、…

mysql 自连接 树形_自连接表的相关问题(树形结构)

问题一&#xff1a;自连接表肯定是相似 Id(PK),parentId(FK),name ….了Id假如是主键的话&#xff0c;parentId就是外键了&#xff0c;可是树形结构肯定是有根节点了&#xff0c;那么根节点的parentId按理说应该是null,原因是根节点没有父节点&#xff0c;可是主键Id是不能为空…

简述python程序的运行原理_谈谈 Python 程序的运行原理

执行 python demo.py 后&#xff0c;将会启动 Python 的解释器&#xff0c;然后将 demo.py 编译成一个字节码对象 PyCodeObject。在 Python 的世界中&#xff0c;一切都是对象&#xff0c;函数也是对象&#xff0c;类型也是对象&#xff0c;类也是对象(类属于自定义的类型&…

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

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

java 上下文加载器_【深入理解Java虚拟机 】线程的上下文类加载器

线程上下文类加载器线程上下文类加载器( Thread Context ClassLoader) 是从JDK1.2 引入的&#xff0c;类Thread 的getContextClassLoader() 与 setContextClassLoader(Classloader var1) 分别用来设置线程的上下文类加载器。如果没有指定线程的上下文的加载器&#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…