synchronizedList通过同步方法保证线程安全,使用synchronized关键字对每个操作加锁,确保单个操作的原子性;但迭代或复合操作需手动同步,否则可能引发并发异常;其性能较低,适用于读多写少、并发不高的场景,高并发下推荐使用CopyOnWriteArrayList。
Java 中 Collections.synchronizedList 方法通过包装原始列表并返回一个线程安全的视图来保证线程安全。它并不是让底层列表本身变得线程安全,而是为所有公开的操作加上同步控制。
基本原理:使用 synchronized 关键字同步方法
该方法返回一个 List 的包装类,这个类中的每个修改或访问方法都使用了 synchronized 关键字进行同步。这意味着每次调用这些方法时,都必须先获取对象的内置锁(monitor lock),从而防止多个线程同时操作列表。
例如:
public E get(int index) {
synchronized (mutex) { return list.get(index); }
}
public E set(int index, E element) {
synchronized (mutex) { return list.set(index, element); }
}
这里的 mutex 是用于同步的对象锁,默认是列表对象本身,也可以由用户指定。
立即学习“Java免费学习笔记(深入)”;
需要注意的迭代问题
虽然单个操作是线程安全的,但复合操作或遍历仍需手动加锁。比如使用迭代器遍历时,必须在同步块中进行,否则可能抛出 ConcurrentModificationException 或出现数据不一致。
正确的遍历方式如下:
List<String> syncList = Collections.synchronizedList(new ArrayList<>());
synchronized (syncList) {
&for (String s : syncList) {
System.out.println(s);
}
}
这是因为迭代过程涉及多个方法调用(hasNext、next),而 synchronizedList 不会对整个迭代过程自动加锁。
性能与适用场景
由于每个方法调用都涉及加锁,synchronizedList 在高并发环境下性能较低。适合读多写少且并发量不高的场景。若需要更高性能,推荐使用 CopyOnWriteArrayList,它适用于读操作远多于写操作的并发场景。
基本上就这些。synchronizedList 提供的是基础级别的线程安全,关键在于理解它的同步机制和使用限制。
评论(已关闭)
评论已关闭