在 java 程序中怎么保证多线程的运行安全?

保证多线程安全需防止共享资源竞争,1. 用synchronized确保方法或代码块互斥执行;2. 使用ReentrantLock实现更灵活的显式加锁;3. volatile保证变量可见性但不保证原子性;4. 优先采用ConcurrentHashMap、AtomicInteger等线程安全类;5. 核心是根据场景选择合适机制并减少共享。

在 Java 程序中保证多线程运行安全,核心是防止多个线程同时访问共享资源导致数据不一致或程序行为异常。以下是几种常用且有效的方法:

使用 synchronized 关键字

synchronized 是 Java 提供的内置锁机制,可确保同一时刻只有一个线程能执行某段代码。

• 修饰实例方法:锁住当前实例对象(this)
• 修饰静态方法:锁住类的 Class 对象
• 修饰代码块:指定加锁对象,粒度更细

例如:

public synchronized void increase() {
    count++;
}

使用 ReentrantLock 显式锁

ReentrantLock 是 java.util.concurrent.locks 包下的可重入锁,功能比 synchronized 更强大,支持公平锁、尝试获取锁、超时获取等。

• 需要手动加锁和释放锁,通常配合 try-finally 使用
• 可避免死锁风险,控制更灵活

示例:

private final ReentrantLock lock = new ReentrantLock();

public void method() {
    lock.lock();
    try {
        // 安全操作共享资源
    } finally {
        lock.unlock();
    }
}

使用 volatile 关键字

volatile 能保证变量的可见性和禁止指令重排序,适用于状态标志位等简单场景。

• 不保证原子性,不能替代锁
• 适合 boolean flag、状态标记等单一变量读写

例如:

private volatile boolean running = true;

使用线程安全的类和容器

Java 提供了大量线程安全的工具类,优先使用这些类代替手动同步。

ConcurrentHashMap:高性能线程安全 Map
CopyOnWriteArrayList:读多写少场景的线程安全 List
AtomicInteger

等原子类:提供无锁的原子操作

例如:

private AtomicInteger count = new AtomicInteger(0);

count.incrementAndGet(); // 原子自增

基本上就这些。关键是根据场景选择合适的同步方式:简单情况用 synchronized 或 volatile,复杂并发用 ReentrantLock,高频读写考虑并发容器和原子类。合理设计,避免共享,才是根本。