Arrays.asList() 都有什么隐藏的陷阱?

Arrays.asList() 方法我们平时开发中一定经常使用,它是将数组转换为List的一种便捷方式,但它有一些潜在的陷阱需要注意。使用的时候需要多多注意呦。

1.不可变性

Arrays.asList() 返回的List是固定大小的,这意味着它不支持对元素的增删操作。任何试图修改大小的操作都会导致UnsupportedOperationException

因为它返回的是 java.util.Arrays.ArrayList 的实例,而不是 java.util.ArrayListArrays.ArrayList 是一个内部类,它基于一个固定大小的数组,并将其包装为 List 接口的实现。因此,它不支持添加或删除元素,只能对现有元素进行修改。

源码贴张图吧,能看到size属性是按照入参的长度给定的:

图片

示例
import java.util.Arrays;
import java.util.List;

public class Main {
    public static void main(String[] args) {
        // 创建一个固定大小的List
        List<String> list = Arrays.asList("a", "b", "c");
        
        // 尝试添加元素
        // list.add("d"); // 这会抛出 UnsupportedOperationException
        
        // 尝试删除元素
        // list.remove(0); // 这会抛出 UnsupportedOperationException
        
        // 尝试清空列表
        // list.clear(); // 这会抛出 UnsupportedOperationException
        
        // 可以修改现有元素
        list.set(0, "A");
        
        System.out.println(list); // 输出 [A, b, c]
    }
}
2.返回的是java.util.Arrays$ArrayList

返回的列表是 java.util.Arrays 的一个私有静态类,而不是标准的 ArrayList 类。这意味着一些 ArrayList 的方法和特性可能不可用。

Arrays.asList() 返回的 ArrayList 和真正的 ArrayList 在某些方面是相似的,但也有一些重要的区别。

相似之处:
  1. 「元素访问」:两者都支持通过索引访问元素。

  2. 「迭代」:可以使用迭代器或增强型 for 循环来遍历它们。

  3. 「尺寸」:都具有 size() 方法来获取列表的大小。

区别:

1.可修改性:

  • Arrays.asList() 返回的列表是固定大小的,不能添加或删除元素,也不能调整大小。

  • 真正的 ArrayList 具有添加、删除和调整大小的方法,因此它是可修改的。

2.添加/删除元素方法:

  • Arrays.asList() 返回的列表不支持添加或删除元素的方法(例如 add()remove() 等),尝试调用这些方法将会抛出 UnsupportedOperationException 异常。

  • 真正的 ArrayList 具有添加、删除和替换元素的方法。

3.修改元素:

  • Arrays.asList() 返回的列表允许修改元素,但不允许改变列表的大小。

  • 真正的 ArrayList 则允许修改元素,并且也可以添加或删除元素。

4.toArray() 方法:

  • Arrays.asList() 返回的列表的 toArray() 方法返回的数组是底层数组的视图,对该数组的修改将反映在列表中。

  • 真正的 ArrayList 的 toArray() 方法则返回一个全新的数组副本,对该数组的修改不会影响到列表。

5.clear() 方法:

  • 真正的 ArrayList 有一个 clear() 方法,用于清空列表中的所有元素,而 Arrays.asList() 返回的列表不支持此方法。

3.数组元素类型必须是引用类型

Arrays.asList() 方法的参数是可变参数(varargs),并且泛型参数类型被推断为数组的类型。因此,如果传递的是基本数据类型的数组,它将被看作是一个对象,而不是数组。因为它的签名是 asList(T... a),其中 T 是泛型类型。这意味着我们可以传递零个或多个参数给这个方法,而不需要显式地创建一个数组来传递。

对于参数类型的推断,Java 中的自动装箱和拆箱会影响到这一点。自动装箱是指基本数据类型可以自动转换为对应的包装类型,而自动拆箱则是指包装类型可以自动转换为基本数据类型。因此,如果传递的是基本数据类型的数组,Java 会将其自动装箱为对应的包装类型的数组。

这也是为什么如果传递的是基本数据类型的数组,它将被看作是一个对象,而不是数组。这里的数组实际上是对象数组,而不是基本数据类型的数组。

示例:
int[] array = {1, 2, 3};
List<int[]> list = Arrays.asList(array); // 注意这里的泛型参数是 int[]
System.out.println(list.size()); // 输出 1,因为传递了一个对象,即整个数组作为一个元素

相反,如果传递的是包装类型的数组,它将会被识别为数组的数组元素:

Integer[] array = {1, 2, 3};
List<Integer> list = Arrays.asList(array); // 注意这里的泛型参数是 Integer
System.out.println(list.size()); // 输出 3,因为传递了三个元素,即数组的每个元素都是列表的一个元素
4.数组内容修改影响List

因为Arrays.asList() 返回的List是基于原始数组的,如果对原始数组进行修改,会影响到返回的List,反之亦然。

示例:
String[] array = {"one", "two", "three"};
List<String> list = Arrays.asList(array);
System.out.println(list);//输出 [one, two, three]
array[0] = "four";
System.out.println(list); // 输出 [four, two, three]
list.set(1, "five");
System.out.println(Arrays.toString(array)); // 输出 [four, five, three]

图片

5.使用ArrayList构造新的可变List

如果需要一个可变的List,最好使用new ArrayList<>(Arrays.asList(array)),这样就可以避免固定大小和不可变性的问题。

示例:
String[] array = {"one", "two", "three"};
List<String> list = new ArrayList<>(Arrays.asList(array));
list.add("four"); // 不会抛出 UnsupportedOperationException

总体而言,使用Arrays.asList()时,要注意其返回的List的特性,以避免潜在的问题。在需要可变性、对List进行增删操作或处理包含null元素的数组时,建议使用new ArrayList<>(Arrays.asList(array))等方式。

<think>我们分析给定的代码片段:List<List<String>>testLists=Arrays.asList(Arrays.asList(stringa="a","b","c"),//这里有问题Arrays.asList("d","e","f"),Arrays.asList("g","h","j"),Arrays.asList("k","l","z"));错误出现在第一行:Arrays.asList(stringa="a","b","c")在Arrays.asList方法中,我们只能直接传递元素,不能像声明变量那样写。这里应该是:Arrays.asList("a","b","c")另外,注意字符串在Java中应该用双引号,而不是单引号(但原问题中单引号是复制过来的,实际应该是双引号?)。但是原问题中字符串使用了单引号,在Java中单引号用于字符(char),而字符串应该用双引号。所以修正后的代码应该是:List<List<String>>testLists=Arrays.asList(Arrays.asList("a","b","c"),Arrays.asList("d","e","f"),Arrays.asList("g","h","j"),Arrays.asList("k","l","z"));但是,原问题中第一行还有一个错误:在方法调用中试图声明变量(stringa="a")是语法错误。因此,主要错误有两个:1.Arrays.asList的参数中使用了变量声明(stringa="a"),这是不允许的。2.字符串使用了单引号(但在问题中我们看到的是单引号,而在Java中字符串必须用双引号)。但是,我们注意到原问题中第一行写的是:`stringa="a"`,这里有两个问题:-类型标识符应该使用大写开头的String,而不是string(Java区分大小写)-在方法调用中不能声明变量。所以正确的写法是去掉变量声明,直接使用字符串字面值,并且使用双引号。同时,其他行中的字符串字面值也用了单引号?例如`Arrays.asList('1','2','3')`在问题中可能是单引号,但在Java中,单引号表示字符(char),而我们需要字符串(String)。但原问题中我们看到的是字符串(比如在之前的例子中,使用的是双引号?),但这里给出的代码片段使用的是单引号。因此,需要将所有的单引号改为双引号(如果元素是字符串的话)。总结错误:第一行:不能有变量声明,应该直接写字符串(并且用双引号)。所有元素:如果元素是字符串,应该使用双引号;如果使用单引号,则会被视为字符,然后自动装箱为Character,而不是String,这会导致类型不匹配。如果保留单引号,那么类型将是Character,而我们需要的是String,因此会编译错误。所以最终的修正应该是:1.移除第一行中的变量声明(包括类型和变量名以及等号),只保留字符串字面值(并使用双引号)。2.确保所有字符串元素都是用双引号。修正后的代码:List<List<String>>testLists=Arrays.asList(Arrays.asList("a","b","c"),Arrays.asList("d","e","f"),Arrays.asList("g","h","j"),Arrays.asList("k","l","z"));注意:原问题中第四行写的是`Arrays.asList("k","l","z")`,这里使用了双引号,但之前的是单引号?问题中给出的代码是混合的?但在问题描述中我们看到的是单引号(在问题中直接写的是单引号,但是Java中字符串不能用单引号)。因此,我们需要将所有单引号改为双引号。但是,原问题中第一行的写法:`stringa="a"`,这个错误很明显,因为Java中类型必须用String(首字母大写),而且在方法调用中不能有变量声明。因此,主要错误是第一行的语法错误。我们给出修正后的代码,并指出错误:错误1:第一行使用了变量声明`stringa="a"`,这是不允许的。应该直接写字符串字面值。错误2:在代码片段中使用了单引号,应该使用双引号(但注意:在问题中,我们看到的是单引号,但实际Java字符串需要双引号)。因此,除了第一行的变量声明错误外,其他行如果使用单引号也是错误。但是,我们注意到在之前的消息中,用户给出的testLists是这样写的:Arrays.asList('1','2','3')这实际上是错误的,因为Java中单引号表示字符,而这里需要字符串。所以整个代码片段的字符串都应该改为双引号。因此,完整修正如下:原错误代码:Arrays.asList(stringa="a","b","c")->错误:变量声明Arrays.asList('d','e','f')->错误:字符应该改为双引号字符串修正后:Arrays.asList("a","b","c"),Arrays.asList("d","e","f"),Arrays.asList("g","h","j"),Arrays.asList("k","l","z")完整代码:List<List<String>>testLists=Arrays.asList(Arrays.asList("a","b","c"),Arrays.asList("d","e","f"),Arrays.asList("g","h","j"),Arrays.asList("k","l","z"));注意:最后一行缺少一个右括号?但是原问题中,整个Arrays.asList有四个元素,并且末尾的括号是匹配的(有四个右括号,每个Arrays.asList一个,外层的Arrays.asList一个)。但是原问题中外层Arrays.asList后面有一个右括号,而内层的每个Arrays.asList都有对应的右括号吗?实际上,每个Arrays.asList的参数列表后面都要有一个右括号,然后外层Arrays.asList的右括号包裹整个参数列表。所以原问题中,除了第一行错误外,整个列表的括号是匹配的(我们看原问题代码,最后一行后面有个括号,而每个内层Arrays.asList的括号都是匹配的)。因此,修正后就可以编译通过了。</think> 在提供的Java代码中,存在两处关键错误: ### 错误1:变量声明语法错误 ```java Arrays.asList(string a = "a", "b", "c") // 错误写法 ``` **问题**:不能在方法参数列表中进行变量声明(`string a = "a"`) **修正**:直接使用字符串字面值 ```java Arrays.asList("a", "b", "c") // 正确写法 ``` ### 错误2:单引号使用错误 ```java Arrays.asList('d', 'e', 'f') // 错误写法 ``` **问题**:单引号 `'` 在Java中表示字符(Char)类型,而我们需要字符串(String)类型 **修正**:使用双引号表示字符串 ```java Arrays.asList("d", "e", "f") // 正确写法 ``` ### 完整修正后的代码 ```java import java.util.*; public class FixedListExample { public static void main(String[] args) { List<List<String>> testLists = Arrays.asList( Arrays.asList("a", "b", "c"), // 修正:移除变量声明 + 使用双引号 Arrays.asList("d", "e", "f"), // 修正:单引号→双引号 Arrays.asList("g", "h", "j"), Arrays.asList("k", "l", "z") ); System.out.println(testLists); } } ``` ### 额外说明 1. 类型大小写:`string` 应为大写开头的 `String`(Java区分大小写) 2. 数据结构:这是创建包含子列表的二维列表结构 3. 自动装箱:单引号创建的Char会被自动装箱为Character类型,与声明的`List<String>`类型不匹配 ### 验证代码输出 ``` [[a, b, c], [d, e, f], [g, h, j], [k, l, z]] ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值