Java高频面试之集合-06

hello啊,各位观众姥爷们!!!本baby今天来报二次道了!哈哈哈哈哈嗝🐶

面试官:ArrayList有哪几种线程安全的实现方式?

在Java中,ArrayList本身是非线程安全的,但在多线程环境下可以通过以下几种方式实现线程安全的操作:


1. 使用 Collections.synchronizedList 包装

通过Collections.synchronizedList()方法将ArrayList包装为线程安全的列表。
实现方式

java 复制代码
List<String> syncList = Collections.synchronizedList(new ArrayList<>());

特点

  • 所有方法(如addremove)通过synchronized同步,保证线程安全。
  • 缺点
    • 高并发时性能较差(锁竞争)。
    • 迭代时需手动同步,否则可能抛出ConcurrentModificationException

示例

java 复制代码
// 写入操作自动同步
syncList.add("value");

// 迭代时需手动同步
synchronized (syncList) {
    Iterator<String> it = syncList.iterator();
    while (it.hasNext()) {
        System.out.println(it.next());
    }
}

2. 使用 CopyOnWriteArrayList

java.util.concurrent.CopyOnWriteArrayList 是专为读多写少场景设计的线程安全列表。
实现方式

java 复制代码
CopyOnWriteArrayList<String> cowList = new CopyOnWriteArrayList<>();

特点

  • 读操作无锁:直接访问底层数组,性能高。
  • 写操作加锁:通过复制新数组实现,保证线程安全。
  • 缺点
    • 写操作开销大(需复制数组),不适合频繁修改的场景。
    • 数据弱一致性(迭代器遍历的是写操作前的快照)。

示例

java 复制代码
cowList.add("value"); // 写操作加锁
cowList.get(0);       // 读操作无锁

3. 使用 Vector(遗留方案)

Vector是Java早期的线程安全列表,通过synchronized修饰所有方法实现同步。
实现方式

java 复制代码
Vector<String> vector = new Vector<>();

特点

  • 方法级同步,线程安全。
  • 缺点
    • 性能差(锁粒度大,高并发时竞争激烈)。
    • 功能落后(如没有Iterator的快速失败机制)。

示例

java 复制代码
vector.add("value"); // 同步方法
vector.get(0);       // 同步方法

4. 手动同步(显式加锁)

通过显式使用synchronizedReentrantLock控制对ArrayList的访问。
实现方式

java 复制代码
List<String> list = new ArrayList<>();
ReentrantLock lock = new ReentrantLock();

// 写入操作
lock.lock();
try {
    list.add("value");
} finally {
    lock.unlock();
}

// 读取操作
lock.lock();
try {
    String value = list.get(0);
} finally {
    lock.unlock();
}

特点

  • 灵活控制锁粒度(如读写分离)。
  • 缺点
    • 代码复杂度高(需手动管理所有访问点)。
    • 容易遗漏同步,导致线程安全问题。

5. 使用并发工具类(如BlockingQueue

对于特定场景(如生产者-消费者模型),可用线程安全的队列代替ArrayList
实现方式

java 复制代码
BlockingQueue<String> queue = new LinkedBlockingQueue<>();

特点

  • 内置阻塞/超时机制,适合任务队列场景。
  • 缺点
    • 功能与List不同(如无随机访问)。

对比与选型建议

方案 线程安全机制 性能 适用场景
Collections.synchronizedList 方法级synchronized 一般 低并发,需兼容现有List接口
CopyOnWriteArrayList 写时复制 读高,写低 读多写少(如配置表、监听器列表)
Vector 方法级synchronized 遗留系统兼容
手动同步 显式锁 灵活 需细粒度控制的场景
BlockingQueue 内部锁/条件变量 高(队列场景) 生产者-消费者模型

🧣

  • 通用场景 :优先选择CopyOnWriteArrayList(读多写少)或Collections.synchronizedList(兼容List接口)。
  • 高并发写入 :考虑手动同步或使用并发队列(如ConcurrentLinkedQueue)。
  • 避免使用Vector:除非需兼容旧代码,其性能和功能已落后于现代并发工具。
相关推荐
WX-bisheyuange2 小时前
基于Spring Boot的教师个人成果管理系统的设计与实现
java·spring boot·后端
xunyan62343 小时前
面向对象(上)-封装性的引入
java·开发语言
脸大是真的好~3 小时前
黑马JAVAWeb-05 JDBC入门-预编译SQL-Mybatis入门-Mybatis日志输出-数据库连接池-增删改查-XML映射配置
java
还算善良_3 小时前
XML签名
xml·java·开发语言
梅梅绵绵冰3 小时前
xml方式实现AOP
xml·java·开发语言
桦说编程4 小时前
Guava 迭代器增强类介绍
java·后端·设计模式
235165 小时前
【JVM】Java为啥能跨平台?JDK/JRE/JVM的关系?
java·开发语言·jvm·spring boot·后端·spring·职场和发展
风轻扬7775 小时前
SQLAlchemy2.0使用
python·sqlalchemy·orm框架
潘帕斯的雄鹰5 小时前
直观理解注意力机制
python·transformer·注意力机制·自注意力机制
打酱油程序员5 小时前
Python数据挖掘详细学习指南
python