当前位置: 首页 > news >正文

Arrays.asList() 返回的list不能add,remove

一.Arrays.asList() 返回的list不能add,remove

Arrays.asList()返回的是List,而且是一个定长的List,所以不能转换为ArrayList,只能转换为AbstractList

原因在于asList()方法返回的是某个数组的列表形式,返回的列表只是数组的另一个视图,而数组本身并没有消失,对列表的任何操作最终都反映在数组上. 所以不支持remove,add方法的

下面是一段很简单的测试代码:  

public class MainFacade {  public static void main(String[] args) {  List<Integer> list = Arrays.asList(1,2,3);  list.add(5);  System.out.print(list.toString());  }  
}  

不过上面的代码会throw出一个UnsupportedOperationException这样的异常  

Exception in thread "main" java.lang.UnsupportedOperationException at java.util.AbstractList.add(AbstractList.java:148) at java.util.AbstractList.add(AbstractList.java:108) at org.popkit.MainFacade.main(MainFacade.java:14) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:606) at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)  


终其原因是Arrays.asList方法返回的ArrayList是继承自AbstractList同时实现
了RandomAccess和Serializable接口,定义如下:  

private static class ArrayList<E> extends AbstractList<E>  implements RandomAccess, java.io.Serializable  

我们再来看看AbstractList这个类的定义:  

public abstract class AbstractList<E> extends AbstractCollection<E> implements List<E>   

这时我们发现AbstractList这个类的set add remove方法定义如下:

public void add(int index, E element) {  throw new UnsupportedOperationException();  
}  public E set(int index, E element) {  throw new UnsupportedOperationException();  
}  public E remove(int index) {  throw new UnsupportedOperationException();  
}  

现在知道了它throw UnsupportedOperationException异常的原因了。  

通过上面的分析,我们知道,其实通过asList方法得到的List是只读的,那么平时我们怎样避免这样的错误发生?我们可以采用如下方法: 

List<Integer> list = new ArrayList<>(Arrays.asList(1,2,3));  

二、Arrays.asList()陷阱

代码如下: 

Java代码  

public static void main(String[] args) {  int[] data = {1,2,3,4,5};  List list = Arrays.asList(data);  System.out.println("列表中的元素数量是:" + list.size());  }  


  注意这里输出的数量是1,原因是,在Arrays.asList中,其接收的参数原型其实是泛型变长参数来的,而基本类型是不能作为范型的参数,按道理应该使用包装类型,但这里缺没有报错, 
因为数组是可以泛型化的,所以转换后在list中就有一个类型为int的数组 

Java代码  

        int[] data = {1,2,3,4,5};  List list = Arrays.asList(data);  System.out.println("元素类型:" + list.get(0).getClass());  System.out.println("前后是否相等:"+data.equals(list.get(0)));  


  可以看到,输出的为元素类型:class [I  
因为jvm不可能输出array类型,array类型属于java.lang.reflect包,通过反射访问 
数组的这个类,编译时候生成的。所以要改为: 

Java代码  

Integer[] data = {1,2,3,4,5};  
List list = Arrays.asList(data);  
System.out.println("列表中的元素数量是:" + list.size());  


 此外,假如如下代码: 

Java代码  

//枚举,声明一个星期  
enum Week{Sun,Mon, Tue, Wed,Thu,Fri,Sat}      
public static void main(String[] args) {  //工作日  Week[] workDays = {Week.Mon, Week.Tue, Week.Wed,Week.Thu,Week.Fri};  //转换为列表  List<Week> list = Arrays.asList(workDays);  //增加周六也为工作日  list.add(Week.Sat);  


 注意这里用add会出错,因为是arrays.aslist中,看代码可以看到这里返回的 
ArrayList不是原来的传统意义上的java.util.arraylist了,而是自己工具类的一个 
静态私有内部类,并没有提供add方法,要自己实现,所以这里是出错了,因此, 
除非确信array.aslist后长度不会增加,否则谨慎使用: 
   List<String> abc=Arrays.asList("a","b","c"),因为这样的长度是无法再add的了 

http://www.lryc.cn/news/120837.html

相关文章:

  • 命令执行漏洞
  • Hive 中 sort by 和 order by 的区别
  • 网络资源利用最大化:爬虫带宽优化解决方案
  • STDF - 基于 Svelte 和 Tailwind CSS 打造的移动 web UI 组件库,Svelte 生态里不可多得的优秀项目
  • C语言一些有趣的冷门知识
  • Oracle数据库审计
  • Node.js新手在哪儿找小项目练手?
  • 全国各城市-货物进出口总额和利用外资-外商直接投资额实际使用额(1999-2020年)
  • CentOS 7查看磁盘空间
  • 基于PHP的轻量级博客typecho
  • MySQL多表查询
  • 消息队列(12) - 定义服务器类
  • 做正确的事 VS 正确的做事,哪个更重要?
  • 每日一题——寻找旋转排序数组中的最小值(I)
  • C语言每日一题:16:数对。
  • 中科亿海微浮点数转换定点数
  • JavaScript激活严格模式
  • Linux cond_resched()简介
  • 初出茅庐的小李博客之认识编码器
  • NVIDIA TX2 NX编译及更新设备树
  • 从零开始学Python(二)运算符、if、循环结构
  • Sentinel整合Spring Cloud Gateway、Zuul详解
  • wsl2安装mysql环境
  • C#质检工具(StyleCop、SonarLint)
  • PyTorch翻译官网教程-NLP FROM SCRATCH: GENERATING NAMES WITH A CHARACTER-LEVEL RNN
  • 【C语言】结构体详解
  • leetcode242. 有效的字母异位词
  • Unity 编辑器资源导入处理函数 OnPostprocessAudio :深入解析与实用案例
  • uniapp开发(由浅到深)
  • QT-基于Buildroot构建系统镜像下实现QT开发