答案:Java中实现线程安全的核心是通过同步机制、原子类、锁和避免共享来保证数据一致性。1. 使用synchronized关键字可对方法或代码块加锁,确保同一时刻只有一个线程执行;2. ReentrantLock提供更灵活的显式锁控制,支持尝试获取、超时和中断,需配合finally释放锁;3. 原子类如AtomicInteger基于CAS实现无锁并发,适用于高并发下的简单变量操作;4. ThreadLocal为每个线程提供独立副本,避免共享竞争,适合线程私有状态管理;选择方式应根据场景权衡性能与复杂度。

在Java中实现线程安全的共享资源访问,核心是确保多个线程对共享数据的操作不会导致数据不一致或程序行为异常。主要手段包括使用同步机制、原子类、锁以及设计无状态或不可变对象。以下是几种常见且有效的实现方式。
使用synchronized关键字
synchronized 是最基础的线程同步手段,可保证同一时刻只有一个线程执行被修饰的代码块或方法。
说明:
它可以修饰实例方法、静态方法或代码块。当用于实例方法时,锁的是当前实例(this);用于静态方法时,锁的是类对象;用于代码块时,可以指定任意对象作为锁。
示例:
public synchronized void increment() {
count++;
}
或者使用同步代码块更细粒度地控制:
立即学习“Java免费学习笔记(深入)”;
public void increment() {
synchronized(this) {
count++;
}
}
使用ReentrantLock显式锁
java.util.concurrent.locks.ReentrantLock 提供了比 synchronized 更灵活的锁机制,支持公平锁、可中断锁、超时获取锁等高级特性。
建议:
- 在需要尝试获取锁而不阻塞时,使用 tryLock() 方法。
- 务必在 finally 块中释放锁,防止死锁。
示例:
private final ReentrantLock lock = new ReentrantLock();
public void increment() {
lock.lock();
try {
count++;
} finally {
lock.unlock();
}
}
使用原子类(Atomic Classes)
java.util.concurrent.atomic 包提供了如 AtomicInteger、AtomicLong、AtomicReference 等原子操作类,底层通过CAS(Compare-And-Swap)实现无锁并发,性能通常优于加锁方式。
适用场景:
- 计数器、状态标志等简单变量的并发更新。
- 高并发环境下对共享变量的读写操作。
示例:
private AtomicInteger count = new AtomicInteger(0);
public void increment() {
count.incrementAndGet();
}
使用ThreadLocal避免共享
有时共享本身就是问题的根源。ThreadLocal 可以为每个线程提供独立的变量副本,从而避免共享资源的竞争。
注意:
适用于每个线程需要独立状态的场景,比如数据库连接、用户会话信息等。使用后注意清理,防止内存泄漏。
示例:
private Static ThreadLocal<Integer> threadLocalCount =
new ThreadLocal<>() {
@Override
protected Integer initialValue() {
return 0;
}
};
基本上就这些。选择哪种方式取决于具体场景:若只是简单同步,synchronized 足够;若需更高性能或更复杂控制,考虑 ReentrantLock 或原子类;若能避免共享,则 ThreadLocal 是不错的选择。关键是理解每种机制的适用边界和开销。


