使用Arrays.asList()的缘由无非是想将数组或一些元素转为集合,而你获得的集合并不必定是你想要的那个集合。java
而一开始asList()的设计时用于打印数组而设计的,但jdk1.5开始,有了另外一个比较更方便的打印函数Arrays.toString(),因而打印再也不使用asList(),而asList()恰巧可用于将数组转为集合。面试
若是你这样使用过,那你要注意下了。数组
将基本类型数组做为asList的参数微信
int[] arr = {1,2,3}; List list = Arrays.asList(arr); System.out.println(list.size());
猜一下输出结果?架构
将数组做为asList参数后,修改数组或Listdom
String[] arr = {"欢迎","关注","Java"}; List list = Arrays.asList(arr); arr[1] = "爱上"; list.set(2,"我"); System.out.println(Arrays.toString(arr)); System.out.println(list.toString());
猜一下输出结果?ide
数组转换为集合后,进行增删元素函数
String[] arr = {"欢迎","关注","Java"}; List list = Arrays.asList(arr); list.add("新增"); list.remove("关注");
猜一下输出结果?工具
你是否是觉得上面👆那个list是 java.util.ArrayList ?ui
答案很肯定:NO!
咱们经过asList()源码可发现,但为了更直观,咱们经过IDEA debug来看看结果。
List<String> asList = Arrays.asList("欢迎","关注","码上实战"); ArrayList<String> aList = new ArrayList<>(asList);
其实它返回的是 java.util.Arrays.ArrayList
,这个家伙是谁呢?
请看下源码:
public class Arrays { //省略其余方法 public static <T> List<T> asList(T... a) { return new ArrayList<>(a); } //就是这个家伙 👇 private static class ArrayList<E> extends AbstractList<E> implements RandomAccess, java.io.Serializable{ private final E[] a; ArrayList(E[] array) { a = Objects.requireNonNull(array); } @Override public int size() { return a.length; } //省略其余方法 } }
但它和ArrayList貌似很像唉!有什么不一样吗?
Arrays.ArrayList 是工具类 Arrays 的一个内部静态类,它没有彻底实现List的方法,而 ArrayList直接实现了List 接口,实现了List全部方法。
Arrays.ArrayList是一个定长集合,由于它没有重写add,remove方法,因此一旦初始化元素后,集合的size就是不可变的。
Arrays.ArrayList将外部数组的引用直接经过“=”赋予内部的泛型数组,因此本质指向同一个数组。
ArrayList(E[] array) { a = array; }
ArrayList是将其余集合转为数组后copy到本身内部的数组的。
public ArrayList(Collection<? extends E> c) { // toArray 底层使用的是 数组clone 或 System.arraycopy elementData = c.toArray(); }
因为Arrays.ArrayList参数为可变长泛型,而基本类型是没法泛型化的,因此它把int[] arr数组当成了一个泛型对象,因此集合中最终只有一个元素arr.
因为asList产生的集合元素是直接引用做为参数的数组,因此当外部数组或集合改变时,数组和集合会同步变化,这在平时咱们编码时可能产生莫名的问题。
因为asList产生的集合并无重写add,remove等方法,因此它会调用父类AbstractList的方法,而父类的方法中抛出的倒是异常信息。
int[] a = {1,2,3}; List list = CollectionUtils.arrayToList(a); System.out.println(list);
int intArray[] = {1, 2, 3}; List<Integer> iList = Arrays.stream(intArray) .boxed() .collect(Collectors.toList()); System.out.println(iList);
Integer intArray[] = {1, 2, 3}; ArrayList<Integer> aList = new ArrayList<>(); for (Integer i: intArray){ aList.add(i); }
显然这种方式不够优雅!反正我不肯意使用。
上面方案不够优雅,那么这种相对来讲优雅一些。
List<String> list = new ArrayList(); Collections.addAll(list, "welcome", "to", "china");
你觉得这种还不错?
too young too simple!
addAll()方法的实现就是用的上面遍历的方式。
既能够用于基本类型也能够返回想要的集合。
int intArray[] = {1, 2, 3}; List<Integer> iList = Arrays.stream(intArray) .boxed() .collect(Collectors.toList()); System.out.println(iList);
将Arrays.asList返回的集合做为ArrayList的构造参数
ArrayList arrayList = new ArrayList<>(Arrays.asList("welcome", "to", "china"));
勿以点小而不闻!体现程序素养或许就在这些小地方,不要给本身或别人留坑。
那么这个知识点,你get到了吗?get到了,那来继续关注我。没get到?来来来,咱俩单独聊聊。
关注微信公众号 「码上实战」 回复 :面试视频 和 架构师 送你很是不错的资料!