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:除非需兼容旧代码,其性能和功能已落后于现代并发工具。
相关推荐
仙人掌_lz4 分钟前
理解多智能体深度确定性策略梯度MADDPG算法:基于python从零实现
python·算法·强化学习·策略梯度·rl
带刺的坐椅10 分钟前
jFinal 使用 SolonMCP 开发 MCP(拥抱新潮流)
java·ai·solon·jfinal·mcp
HsuHeinrich28 分钟前
利用散点图探索宇航员特征与太空任务之间的关系
python·数据可视化
陌尘(MoCheeen)31 分钟前
技术书籍推荐(002)
java·javascript·c++·python·go
牛马baby32 分钟前
Java高频面试之并发编程-16
java·开发语言·面试
cainiao08060532 分钟前
《Spring Boot 4.0新特性深度解析》
java·spring boot·后端
满怀101534 分钟前
【人工智能核心技术全景解读】从机器学习到深度学习实战
人工智能·python·深度学习·机器学习·tensorflow
乐言36139 分钟前
Jmeter中的BeanShell如何使用?
python·jmeter·压力测试
zizisuo43 分钟前
面试篇:Spring MVC
java·spring·mvc
-曾牛44 分钟前
Spring AI 与 Hugging Face 深度集成:打造高效文本生成应用
java·人工智能·后端·spring·搜索引擎·springai·deepseek