避免Arrays.asList陷阱:优雅处理结构性修改的方法

news/2024/11/23 9:58:31/

临近年终,项目交付排期比较紧张,导致很多时候,Code Review 往往是走马观花,没有严格执行。最近,一个实习生就产生了一个十分低级的代码BUG。笔者感觉这个问题,对于实习生,尤其是刚入职的 应届 Java 开发工程师具有一定的普遍性。因此和大家分享一下。

问题背景

项目中有个配置,默认是3个属性值。要求开发先读取这3个配置,然后从数据库中提取所有的数据,如果数据库中的数据和默认配置不同的话,就要增加上。很自然的,我们的实习生执行了下面的判断逻辑。

public static String[] INFO = new String[]{"weibo_name", "weibo_age"};
public static List<WeiInfoBean> getAllWeiboInfo(String str) {List<String> infoList = Arrays.asList(INFO);//...代码逻辑省略infoList.add(((JSONObject) itemJson).getString("item_name"));return list;
}

在UAT环境的时候,因为配置不全,因此 getAllWeiboInfo()函数没有执行。在后续的测试中,也没有细究这方面的代码。但是上线后,系统直接抛出异常:

Exception in thread "main" java.lang.UnsupportedOperationExceptionat java.util.AbstractList.add(AbstractList.java:148)at java.util.AbstractList.add(AbstractList.java:108)at com.weibo.service.impl.WeiboServiceImpl.getAllWeiboInfo(WeiboServiceImpl.java:122)

排查过程

系统报警后,笔者第一时间接手代码,并且迅速定位到了问题的所在:实习的开发同事错误的使用了Arrays.asList()函数,并且进行了增删操作。

在将对应的代码修复后,系统恢复正常。

问题分析

在Java编程中,Arrays.asList方法是一个常见的工具,用于将数组转换为List集合。然而,在使用这个方法的过程中,有时会遇到一些报错,其中一个比较常见的是在尝试向Arrays.asList返回的List中添加元素时抛出UnsupportedOperationException异常。

Arrays.asList方法简介

Arrays.asList是Java中Arrays类提供的一个静态方法,它的主要作用是将数组转换为List。该方法接受一个数组作为参数,并返回一个包装了该数组的固定大小的List。以下是Arrays.asList方法的基本用法:

String[] array = {"A", "B", "C"};
List<String> list = Arrays.asList(array);

在上述示例中,数组array被转换为一个包含相同元素的List集合list

Arrays.asList的限制

尽管Arrays.asList方法提供了方便的数组转换功能,但它有一个重要的限制:返回的List是一个固定大小的List。这意味着,通过Arrays.asList返回的List对象,不能进行结构上的修改操作,例如添加、删除元素等。如果尝试对这个List执行结构上的修改操作,就会触发UnsupportedOperationException异常。

UnsupportedOperationException异常的原因

在运行时,当尝试向由Arrays.asList返回的List添加或删除元素时,会抛出UnsupportedOperationException异常。这是因为Arrays.asList返回的List是一个只读的视图,它直接映射到原始数组,不允许对其进行结构上的修改。

String[] array = {"A", "B", "C"};
List<String> list = Arrays.asList(array);// 尝试添加元素
list.add("D"); // 抛出UnsupportedOperationException异常

在这个例子中,由于Arrays.asList返回的List是不可变的,试图添加元素"D"导致了UnsupportedOperationException异常。

源码分析

我们直接打开JDK源码进行分析

 public static <T> List<T> asList(T... a) {return new ArrayList<>(a);}// ArrayList的构造函数
public ArrayList(E[] array) {this.a = Objects.requireNonNull(array);
}

在构造ArrayList时,传入的数组将直接被引用,而ArrayList的实现并没有提供对数组进行修改的方法。因此,当尝试在返回的List上执行add、remove等修改操作时,就会抛出UnsupportedOperationException异常。

其他使用中的问题

除了增删操作会导致系统报错外,Arrays.asList 方法还有其他使用上的问题,如果不加以注意,就会导致结果和想得到的不同。这里一起提出来,方便大家警醒

不要用基础数据类型转换数组

我们直接使用代码作为说明:

 public static void main(String[] args) {int[] in = new int[]{1,2,3};List list =Arrays.asList(in);System.out.println(list.size());// 结果是1}

上述代码的结果为1,这是为什么呢?我们还是看一下源码:

  @SafeVarargs@SuppressWarnings("varargs")public static <T> List<T> asList(T... a) {return new ArrayList<>(a);}

在源码里,Arrays.asList方法的工作原理是将传入的数组作为参数,创建一个包装该数组的List。但是,由于Java泛型不支持基础数据类型,因此对于基础类型数组,它会将整个数组作为单一元素添加到List中,而不是将数组的每个元素都添加为List的单独元素。

我们需要修改成:

int[] intArray = {1, 2, 3, 4, 5};
List<int[]> intList = Arrays.asList(intArray);

原数组修改,会导致list同步修改

我们直接使用代码作为说明:

public static void main(String[] args) {int[] in = new int[]{1,2,3};List list =Arrays.asList(in);in[0] = 4;System.out.println(list.get(0));// 结果是4}

在使用 Arrays.asList 转换数组为列表时,生成的列表其实是基于原始数组的视图,而不是新的独立副本。这就意味着,如果修改原数组,会影响到由 Arrays.asList 返回的列表,反之亦然。

这种现象的原因在于 Arrays.asList 返回的列表是一个包装了原始数组的固定大小的列表,它直接引用原始数组,而不是复制数组元素到新的列表中。因此,对原数组的修改会在列表中反映出来,反之亦然。

如何优雅使用 Arrays.asList 方法

使用ArrayList构造新的可修改List

String[] array = {"A", "B", "C"};
List<String> list = new ArrayList<>(Arrays.asList(array));
list.add("D"); // 正常执行

通过使用ArrayList的构造函数,将Arrays.asList返回的List转换为一个新的ArrayList,就可以避免UnsupportedOperationException异常,从而可以执行结构上的修改操作。

使用Collections.unmodifiableList方法

String[] array = {"A", "B", "C"};
List<String> list = Collections.unmodifiableList(Arrays.asList(array));// 尝试添加元素
list.add("D"); // 抛出UnsupportedOperationException异常

通过使用Collections.unmodifiableList方法,可以将Arrays.asList返回的List包装为一个不可修改的List,从而禁止对其进行结构上的修改。

总结

Arrays.asList是一个在Java中方便的数组转换工具,是一个经常被开发者使用的方法。但需要注意返回的List是固定大小且不可修改的。在处理需要修改List结构的情况下,使用ArrayList构造新的List或者使用Collections.unmodifiableList进行包装是解决问题的常见方法。通过深入理解Arrays.asList方法的特性,开发者可以更好地利用这个工具,并避免常见的异常情况。通过源码分析,我们更清晰地理解了为什么Arrays.asList返回的List是不可修改的,以及如何避免相关异常。


http://www.ppmy.cn/news/1345868.html

相关文章

docker下拉(pull)镜像和生成容器,文章尾部有常用的linux命令

目录 1&#xff1a;docker镜像和容器是什么 2&#xff1a;docker初始化个容器&#xff0c;并进入容器安装mariaDb和httpd 1&#xff1a;用远程工具SecureCRT登录docker 2&#xff1a;拉取CentOS镜像并初始化一个容器 a&#xff1a;拉取镜像&#xff08;这一步可能会有点久&…

QT QDialog 中的按钮,如何按下后触发 accepted 消息?

QT 作为跨平台的系统&#xff0c;对话框并没有采用 Windows API 那种模式&#xff0c;通过返回 mrOK、mrCancel 等结果告诉调用方结果&#xff0c;而是采用了 accepted、rejected 等信号确定执行结果。下面介绍几种出发这些信号的方法。 1. 在按钮的 clicked 槽函数中触发 acc…

Gazebo的初始启动问题

在机器人开发之中一般初始启动会输入以下语句&#xff1a; ros2 launch gazebo_ros gazebo.launch.py 通常都会报错&#xff0c;原因是路径并未添加&#xff0c;输入下列语句到.bashrc即可 source /usr/share/gazebo/setup.bash

计算机网络相关题目及答案(第三章)

第三章 复习题: R3.考虑在主机A和主机B之间有一条TCP连接。假设从主机A传送到主机B的TCP报文段具有源端口 号x和目的端口号y。对于从主机B传送到主机A的报文段,源端口号和目的端口号分别是多少? 答:源端口号将设置为y, 目的端口号设置为x. 因为在传输UDP包的时候, 网络…

MySQL数据库⑤_基本查询DQL_表的增删查改DML

目录 1. CRUD介绍 2. Create 新增 2.1 单行数据全列插入 2.2 多行数据指定列插入 2.3 插入否则更新 2.4 替换数据 3. Retrieve 查找 3.1 select 查询 3.2 where 条件 3.2.1 MySQL运算符 3.2.2 NULL的查询 3.3 order by 结果排序 3.4 limit 筛选分页结果 4. Updat…

【Spring】Spring 对 Ioc 的实现

一、Ioc 控制反转 控制反转是一种思想 控制反转是为了降低程序耦合度&#xff0c;提高程序扩展力&#xff0c;达到 OCP 原则&#xff0c;达到 DIP 原则 控制反转&#xff0c;反转的是什么&#xff1f; 将对象的创建权利交出去&#xff0c;交给第三方容器负责 将对象和对象之…

IP地址如何保护网络安全

面对网络攻击时&#xff0c;仅依靠常态化的网络安全防御系统已捉襟见肘&#xff0c;如联合使用IP地址数据可以形成多元化的安全解决方案&#xff0c;全面监控网络活动&#xff0c;发现潜在威胁&#xff0c;制定有针对性的应对措施。 网络攻击追踪 当网站或应用遭受DDoS等网络攻…

新版MQL语言程序设计:命令模式的原理、应用及代码实现

文章目录 一、什么是命令模式二、命令模式的实现原理三、命令模式的应用场景四、命令模式的代码实现 一、什么是命令模式 命令模式是一种行为设计模式&#xff0c;它将请求封装成一个对象&#xff0c;从而使你可以用不同的请求对客户端进行参数化。这种模式的主要目的是将方法的…