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:除非需兼容旧代码,其性能和功能已落后于现代并发工具。
相关推荐
weixin_4723394624 分钟前
高效处理大体积Excel文件的Java技术方案解析
java·开发语言·excel
小毛驴8501 小时前
Linux 后台启动java jar 程序 nohup java -jar
java·linux·jar
DKPT1 小时前
Java桥接模式实现方式与测试方法
java·笔记·学习·设计模式·桥接模式
好奇的菜鸟3 小时前
如何在IntelliJ IDEA中设置数据库连接全局共享
java·数据库·intellij-idea
DuelCode4 小时前
Windows VMWare Centos Docker部署Springboot 应用实现文件上传返回文件http链接
java·spring boot·mysql·nginx·docker·centos·mybatis
优创学社24 小时前
基于springboot的社区生鲜团购系统
java·spring boot·后端
幽络源小助理4 小时前
SpringBoot基于Mysql的商业辅助决策系统设计与实现
java·vue.js·spring boot·后端·mysql·spring
猴哥源码4 小时前
基于Java+springboot 的车险理赔信息管理系统
java·spring boot
烛阴5 小时前
简单入门Python装饰器
前端·python
YuTaoShao5 小时前
【LeetCode 热题 100】48. 旋转图像——转置+水平翻转
java·算法·leetcode·职场和发展