List
是Java集合框架中一个非常重要的接口,它代表了一个有序的集合,允许元素重复,并且可以按照插入的顺序进行访问。
我们先来看看List在集合中的位置:
List
是单列集合接口Collection
下的一个分支,另两个分支是Set
和Queue
,三者的区别:
List接口继承自Collection接口,位于java.util包中。
List是有序集合的抽象表示,Java发展至今,List体系已经非常庞杂。
从上图看出,JDK中,直接或间接继承List接口的有80个类。当然,我们无需一一学习,通过三个常用的实现类的学习掌握原理即可:ArrayList、LinkedList和Vector。
ArrayList
:基于动态数组实现,提供了快速的随机访问。LinkedList
:基于双向链表实现,擅长插入和删除操作,尤其是表头和表尾的操作。Vector
:早期版本的线程安全列表,与ArrayList相似,但现在多被ArrayList取代,因同步开销较大。import java.util.*;
public class ListDemo {
public static void main(String[] args) {
List<String> arrayList = new ArrayList<>(); // 创建ArrayList实例
List<String> linkedList = new LinkedList<>(); // 创建LinkedList实例
}
}
add(E element)
:在列表末尾添加元素。add(int index, E element)
:在指定位置插入元素。arrayList.add("Apple");
linkedList.add(0, "Banana"); // 在首位插入
get(int index)
System.out.println(arrayList.get(0));
for (String fruit : arrayList) {
System.out.println(fruit);
}
set(int index, E element)
方法替换指定位置的元素。arrayList.set(0, "Orange");
remove(int index)
:根据索引删除。remove(Object o)
:根据元素删除第一个匹配项。arrayList.remove(0);
arrayList.remove("Orange");
contains(Object o)
:判断是否包含某元素。indexOf(Object o)
:返回元素第一次出现的索引,未找到返回-1。lastIndexOf(Object o)
:List集合中的元素可重复,返回元素最后一次出现的索引,未找到返回-1。boolean hasApple = arrayList.contains("Apple");
int index = arrayList.indexOf("Apple");
int index = arrayList.lastIndexOf("Apple");
size()
:返回列表大小。clear()
:清空列表。int size = arrayList.size();
arrayList.clear();
Java List 接口提供了多种方式来遍历其中的元素,以下是三种常见的遍历方式,每种方式都有相应的代码示例。
这是最简洁也是最常用的遍历方式,适用于Java 5及以后的版本。通过for-each循环,可以直接遍历List中的每个元素,而无需手动管理索引。
代码示例:
import java.util.ArrayList;
import java.util.List;
public class ListTraversal {
public static void main(String[] args) {
List<String> fruits = new ArrayList<>();
fruits.add("Apple");
fruits.add("Banana");
fruits.add("Cherry");
// 使用for-each循环遍历
for (String fruit : fruits) {
System.out.println(fruit);
}
}
}
ConcurrentModificationException
**注意,**使用for
循环时,有可能会出现并发修改异常ConcurrentModificationException
,如下面的例子,假设你有一个任务是遍历一个List,检查其中的元素,如果满足某个条件,就从List中删除该元素。
List<String> list = new ArrayList<>(Arrays.asList("A", "B", "C", "D"));
for (String item : list) {
if ("B".equals(item)) {
list.remove(item); // 这里会抛出ConcurrentModificationException
}
}
Java集合框架中的许多类,如ArrayList
和LinkedList
,为了检测到并发修改,使用了所谓的“快速失败”机制:
modCount
的字段来记录集合的修改次数add
或remove
方法)发生修改时,modCount
就会递增next
或hasNext
等方法时,都会检查这个计数器是否发生变化,如果发现modCount
不等于它内部记录的初始修改次数,就会抛出ConcurrentModificationException
据此分析,上面的代码示例之所以会报错,是因为:
正确的做法是在迭代过程中使用迭代器的remove
方法来删除元素,因为迭代器的remove
方法会在删除元素后同时更新内部的修改计数,以保持一致性。
Iterator<String> iterator = list.iterator();
while (iterator.hasNext()) {
String item = iterator.next();
if ("B".equals(item)) {
iterator.remove(); // 正确的删除方式
}
}
对于多线程环境下的并发修改问题,可以考虑使用CopyOnWriteArrayList
。这是一种线程安全的List实现,它通过在每次修改时创建集合的副本来避免并发修改异常,适合读多写少的场景。
List<String> list = new CopyOnWriteArrayList<>(Arrays.asList("A", "B", "C", "D"));
list.removeIf("B"::equals); // 线程安全的删除操作
迭代器是一种更通用的遍历集合的方法,适用于所有实现了Iterable接口的集合,包括List。通过调用List的iterator()
方法获取Iterator对象,然后使用hasNext()
和next()
方法进行遍历。
代码示例:
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
public class ListTraversal {
public static void main(String[] args) {
List<String> fruits = new ArrayList<>();
fruits.add("Apple");
fruits.add("Banana");
fruits.add("Cherry");
// 使用迭代器遍历
Iterator<String> iterator = fruits.iterator();
while (iterator.hasNext()) {
String fruit = iterator.next();
System.out.println(fruit);
}
}
}
从Java 8开始,可以使用Stream API来遍历和处理集合中的元素,这种方式更加灵活,支持函数式编程风格。
代码示例:
import java.util.ArrayList;
import java.util.List;
public class ListTraversal {
public static void main(String[] args) {
List<String> fruits = new ArrayList<>();
fruits.add("Apple");
fruits.add("Banana");
fruits.add("Cherry");
// 使用Stream API遍历
fruits.stream().forEach(System.out::println);
}
}
在Java中,对List集合中的元素进行排序可以通过多种方式实现,主要依赖于java.util.Collections
类和List接口本身提供的排序方法。下面我将介绍几种常见的排序方法,并提供相应的代码示例。
这是最直接的方式,适用于实现了Comparable
接口的元素列表,进行自然排序。
代码示例(自然排序):
import java.util.*;
class Person implements Comparable<Person> {
String name;
int age;
Person(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public int compareTo(Person other) {
return Integer.compare(this.age, other.age); // 按年龄排序
}
@Override
public String toString() {
return name + " " + age;
}
}
public class SortListExample {
public static void main(String[] args) {
List<Person> people = new ArrayList<>();
people.add(new Person("Tom", 25));
people.add(new Person("Jerry", 20));
people.add(new Person("Bob", 30));
Collections.sort(people);
for (Person person : people) {
System.out.println(person);
}
}
}
如果列表中的元素没有实现Comparable
接口,或者你想根据不同的规则进行排序,可以提供一个Comparator
。
代码示例(自定义比较器排序):
import java.util.*;
public class SortListExample {
public static void main(String[] args) {
List<String> names = Arrays.asList("Alice", "Bob", "Charlie");
Collections.sort(names, new Comparator<String>() {
@Override
public int compare(String s1, String s2) {
return s2.compareTo(s1); // 倒序排序
}
});
names.forEach(System.out::println); // 输出:Charlie, Bob, Alice
}
}
从Java 8开始,List接口直接提供了sort()
方法,它同样接受Comparator来控制排序逻辑。
代码示例:
import java.util.*;
public class SortListExample {
public static void main(String[] args) {
List<Integer> numbers = Arrays.asList(3, 1, 4, 1, 5, 9);
numbers.sort(Integer::compareTo); // 自然排序
System.out.println(numbers); // 输出:[1, 1, 3, 4, 5, 9]
numbers.sort(Collections.reverseOrder()); // 倒序排序
System.out.println(numbers); // 输出:[9, 5, 4, 3, 1, 1]
}
}
Collections.sort()
适用于不支持Lambda表达式的较早Java版本。sort()
方法配合Lambda表达式或方法引用来实现排序更为简洁。Collections.synchronizedList(List<T> list)
或CopyOnWriteArrayList
作为替代。通过上述内容,我们不仅了解了List接口的继承体系、常用操作,还深入探讨了ArrayList、LinkedList和Vector这三种常见实现的底层原理及其应用场景。掌握这些知识,将有助于在实际开发中更加高效、灵活地使用List集合。
因篇幅问题不能全部显示,请点此查看更多更全内容