java.util.List.toArray() 使用体会

下面的代码执行后:

[java]  view plain copy
  1. private void onButtonListToArrayClick() {  
  2.     // Array--->List  
  3.     String[] strArray = new String[] { "aa""bb" };  
  4.     List<String> list = new ArrayList<String>(Arrays.asList(strArray));  
  5.     list.add("cc");  
  6.     // List--->Array  
  7.     String[] strArray2;  
  8.     try {  
  9.         strArray2 = (String[]) list.toArray();  
  10.     } catch (Exception e) {  
  11.         Log.i("ToArray""failure:" + e.getMessage());  
  12.     }  
  13.   
  14. }  

异常发生


原因为list.toArray()返回的是Object类型,强制转换的时候发生ClassCastException

程序修改如下:

[java]  view plain copy
  1. private void onButtonListToArrayClick() {  
  2.     // Array--->List  
  3.     String[] strArray = new String[] { "aa""bb" };  
  4.     List<String> list = new ArrayList<String>(Arrays.asList(strArray));  
  5.     list.add("cc");  
  6.     // List--->Array  
  7.     String[] strArray2 = null;  
  8.     try {  
  9.         strArray2 = list.toArray(new String[0]);  
  10.     } catch (Exception e) {  
  11.         Log.i("ToArray""failure:" + e.getMessage());  
  12.     }  
  13.     for (String element : strArray2) {  
  14.         Log.i("ToArray""strArray2:" + element);  
  15.     }  
  16.   
  17. }  

运行结果:


上面的方法

[java]  view plain copy
  1. strArray2 = list.toArray(new String[0]);  

与下面的方法一样

[java]  view plain copy
  1. strArray2 = new String[list.size()];  
  2. strArray2 = list.toArray(strArray2);  

备注:

下面是ArrayList的两个toArray()方法的源代码:

[java]  view plain copy
  1. public Object[] toArray() {   
  2.    Object[] result = new Object[size];     
  3.    System.arraycopy(elementData, 0, result, 0, size);     
  4.    return result;   
  5. }   
  6. public Object[] toArray(Object a[]) {     
  7. if (a.length < size)         
  8. a = (Object[])java.lang.reflect.Array.newInstance(a.getClass().getComponentType(), size);       System.arraycopy(elementData, 0, a, 0, size);     
  9. if (a.length > size)         
  10. a[size] = null;     
  11. return a;   
  12. }   

①不带参数的toArray方法,是构造的一个Object数组,然后进行数据拷贝,此时进行转型就会产生ClassCastException 

②带参数的toArray方法,则是根据参数数组的类型,构造了一个对应类型的,长度跟ArrayList的size一致的空数组,虽然方法本身还是以 Object数组的形式返回结果,不过由于构造数组使用的ComponentType跟需要转型的ComponentType一致,就不会产生转型异常

转自:http://blog.youkuaiyun.com/gh102/article/details/7301051

-------------------------------------------------------------------------------------------------------------------------

总结:1. 数组的定义和初始化是可以分开的,不能因为经常见到的数组的定义和初始化连在一起操作就混淆,定义如下:

数据类型【】数组名 = new 数据类型 【数组长度】;

  2. 此博主的博文条理清晰明了,有理有据,是小白学习的好棒样(⊙o⊙)哦~~~

<think>我们正在讨论java.util.Set接口。用户之前询问了接口中是否可以包含具体方法,现在转向Set的用法和示例。根据引用[2],Set是Java集合框架中的接口,表示不包含重复元素的集合。它继承自Collection接口,主要特点是:不允许重复元素,最多一个null元素,且通常不保证顺序(但LinkedHashSet和TreeSet等实现类有特定顺序)。我们将重点介绍Set的常用方法、实现类以及使用示例。注意:用户要求用法、示例、方法,所以我们需要覆盖这些点。引用[1]和引用[2]中,引用[2]直接与Set相关,因此我们将主要参考引用[2],同时引用[1]可能与正则表达式相关,但当前问题不涉及,所以可以忽略。回答结构:1.Set接口概述(重复上述关键点)2.常用方法(继承自Collection,但有一些特性)3.常用实现类(HashSet,LinkedHashSet,TreeSet)及特点4.使用示例(代码示例展示常用操作)5.相关问题生成注意:在代码示例中,我们将使用Java语言,并尽量展示主要方法。根据引用[2],Set不允许重复,当我们尝试添加重复元素时,add方法会返回false。我们将按步骤编写回答。</think> ### `java.util.Set` 接口详解 `java.util.Set` 是 Java 集合框架的核心接口,表示**不包含重复元素**的集合(数学集合的实现)。它继承自 `Collection` 接口,具有以下核心特性: 1. **元素唯一性**:禁止重复元素(依赖 `equals()` 判断) 2. **无序性**(大部分实现):除 `LinkedHashSet` 和 `TreeSet` 外不保证顺序 3. **允许 null 元素**:最多包含一个 null 4. **非线程安全**:需手动同步 #### 📌 常用实现类 | 实现类 | 数据结构 | 顺序保证 | 时间复杂度 | 适用场景 | |-----------------|--------------|----------------|------------|------------------------------| | `HashSet` | 哈希表 | ❌ 无 | O(1) | 快速查找、去重 | | `LinkedHashSet` | 哈希表+链表 | ✅ 插入顺序 | O(1) | 需要保留插入顺序的去重 | | `TreeSet` | 红黑树 | ✅ 自然排序 | O(log n) | 需要排序或范围查询的去重 | #### ⚙️ 核心方法(部分继承自 Collection) ```java // 基础操作 boolean add(E e) // 添加元素(重复返回false) boolean remove(Object o) // 删除元素 boolean contains(Object o) // 判断存在 // 批量操作 boolean addAll(Collection<? extends E> c) // 并集 boolean retainAll(Collection<?> c) // 交集 boolean removeAll(Collection<?> c) // 差集 // 工具方法 int size() // 元素数量 void clear() // 清空集合 boolean isEmpty() // 判空 Object[] toArray() // 转为数组 ``` #### ✨ 使用示例 ```java import java.util.*; public class SetExample { public static void main(String[] args) { // 创建HashSet(去重+无序) Set<String> cities = new HashSet<>(); Collections.addAll(cities, "北京", "上海", "广州", "北京"); // 自动去重 System.out.println(cities); // 可能输出 [上海, 北京, 广州] // 交集操作 Set<String> easternCities = Set.of("上海", "杭州", "北京"); cities.retainAll(easternCities); System.out.println("华东城市: " + cities); // 输出 [上海, 北京] // TreeSet排序 Set<Integer> numbers = new TreeSet<>(); Collections.addAll(numbers, 5, 2, 8, 2); System.out.println("排序数字: " + numbers); // 输出 [2, 5, 8] } } ``` #### ⚠️ 注意事项 1. **元素相等判定**:依赖 `equals()` 和 `hashCode()`(需同时重写) 2. **可变对象风险**:修改已存储对象的属性会导致集合状态不一致 3. **线程安全方案**: ```java Set<Object> safeSet = Collections.synchronizedSet(new HashSet<>()); ``` 4. **性能选择**: - 快速查找:`HashSet` - 按插入顺序访问:`LinkedHashSet` - 范围查询/排序:`TreeSet` --- ### 📚 相关问题 1. `Set` 如何判断元素重复?`equals()` 与 `hashCode()` 的协作机制是怎样的? 2. 对比 `HashSet`, `LinkedHashSet`, `TreeSet` 在内存占用和性能上的具体差异? 3. 如何在 `Set` 中存储自定义对象并保证正确去重?实现示例 4. `Set` 与其他集合(如 `List`, `Map`)的主要区别是什么?适用场景对比 5. `java.util.concurrent` 包中提供了哪些线程安全的 `Set` 实现?如何选择?[^2]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值