在Java中,实现线程同步的目的是为了防止多个线程在访问共享资源时产生冲突,保证数据的一致性和完整性。以下是几种常见的线程同步方法:
-
synchronized关键字:
- 同步方法:在方法声明上使用
synchronized
关键字,这样整个方法都成为同步代码块。当一个线程进入同步方法时,会获取到对象的监视器(monitor),其他线程必须等待,直到当前线程退出该方法。public synchronized void synchronizedMethod() { // 同步代码块 }
- 同步代码块:在代码块上使用
synchronized
关键字,可以更细粒度地控制同步范围,锁定的是指定对象的监视器。public void method() { synchronized (object) { // 同步代码块 } }
- 同步方法:在方法声明上使用
-
Lock接口及其实现类(来自
java.util.concurrent.locks
包):ReentrantLock
是Lock接口的一个可重入实现,相比synchronized
,它提供了更多的灵活性,如尝试锁定、定时锁定以及公平锁和非公平锁的选择。private final ReentrantLock lock = new ReentrantLock(); public void method() { lock.lock(); try { // 同步代码块 } finally { lock.unlock(); // 确保锁被释放 } }
-
原子类(
java.util.concurrent.atomic
包):- 对于基本数据类型的简单操作,可以使用原子类(如
AtomicInteger
、AtomicBoolean
等)来避免同步,因为它们使用CAS(Compare and Swap)操作来保证原子性,从而减少了同步的开销。private AtomicInteger counter = new AtomicInteger(0); public void increment() { counter.incrementAndGet(); }
- 对于基本数据类型的简单操作,可以使用原子类(如
-
volatile关键字:
- 虽然
volatile
不是一种同步机制,但它可以确保变量的可见性,即当一个线程修改了volatile变量的值,其他线程可以立刻看到最新值。但volatile不能保证原子性,一般用于状态标记或者配合其他同步机制使用。private volatile boolean flag = false; public void setFlag(boolean value) { flag = value; }
- 虽然
-
并发工具类(
java.util.concurrent
包):- 包括
CountDownLatch
、CyclicBarrier
、Semaphore
、Phaser
等,这些高级并发工具提供了更复杂的同步机制,可以解决特定的并发控制问题。
- 包括
根据不同的应用场景和需求,可以选择合适的同步策略来保证线程安全。