Java的锁机制详解

news/2024/10/9 8:48:53/

在并发编程中, 是用于控制多个线程对共享资源进行访问的工具。Java提供了多种锁机制,从最基础的 synchronized 到高级的 ReentrantLock,这些锁帮助我们确保线程安全,并能有效避免数据竞争和死锁问题。
在这里插入图片描述

1. synchronized 关键字

synchronized 是Java中最简单的锁机制。它可以锁住方法或者代码块,确保某个线程在访问共享资源时,其他线程无法访问同一个资源。

示例代码:同步方法

java">public class SynchronizedExample {public synchronized void synchronizedMethod() {System.out.println("Thread " + Thread.currentThread().getName() + " is executing synchronized method.");}
}

在上述代码中,synchronizedMethod 方法被 synchronized 关键字修饰,意味着同一时间只能有一个线程执行该方法。

示例代码:同步代码块

java">public class SynchronizedBlockExample {private final Object lock = new Object();public void synchronizedBlock() {synchronized (lock) {System.out.println("Thread " + Thread.currentThread().getName() + " is executing synchronized block.");}}
}

使用同步代码块可以灵活地锁定某个对象(如上例中的 lock),只锁住需要保护的部分,而不是整个方法。

锁的粒度问题

synchronized 锁的粒度较粗,可能会导致性能瓶颈。对于更复杂的并发场景,ReentrantLock 等更灵活的锁机制是更好的选择。

2. ReentrantLock

ReentrantLock 是 Java java.util.concurrent.locks 包中的高级锁,它提供了更丰富的功能,如:公平锁可重入性可中断锁 等。

2.1 基本使用

java">import java.util.concurrent.locks.ReentrantLock;public class ReentrantLockExample {private final ReentrantLock lock = new ReentrantLock();public void execute() {lock.lock(); // 获取锁try {System.out.println("Thread " + Thread.currentThread().getName() + " is executing.");} finally {lock.unlock(); // 释放锁}}
}

在这个例子中,我们显式地调用 lock.lock() 来获取锁,并在 finally 中确保锁会被释放,以防止因异常导致死锁。

2.2 公平锁

默认情况下,ReentrantLock 是非公平锁,即等待时间长的线程不一定优先获得锁。你可以通过构造函数指定是否为公平锁:

java">ReentrantLock fairLock = new ReentrantLock(true); // 公平锁

公平锁确保先请求锁的线程先获得锁,但相对性能较低。

3. ReadWriteLock

ReadWriteLock 是一种更细粒度的锁,它允许多个读线程同时访问共享资源,但在有写操作时,写线程会独占资源。典型的实现是 ReentrantReadWriteLock

示例代码

java">import java.util.concurrent.locks.ReentrantReadWriteLock;public class ReadWriteLockExample {private final ReentrantReadWriteLock rwLock = new ReentrantReadWriteLock();public void read() {rwLock.readLock().lock();try {System.out.println("Thread " + Thread.currentThread().getName() + " is reading.");} finally {rwLock.readLock().unlock();}}public void write() {rwLock.writeLock().lock();try {System.out.println("Thread " + Thread.currentThread().getName() + " is writing.");} finally {rwLock.writeLock().unlock();}}
}

这种锁的优势在于读操作不互斥,当多个线程只读数据时,能够提高系统的吞吐量。但在写操作时,所有的读操作会被阻塞,确保数据的一致性

4. 锁优化技术

4.1 偏向锁

偏向锁 是Java的轻量级锁优化策略。它假定大多数情况下锁不会被多个线程竞争,因此,第一次获取锁时,锁会偏向获取锁的线程。如果没有其他线程竞争,这个线程之后获取锁的代价几乎为零。

偏向锁适用于线程独占资源的场景,而不适用于高并发的竞争场景。

4.2 自旋锁

自旋锁 是通过让线程循环等待一段时间,而不立即进入阻塞状态,来减少上下文切换的开销。在高并发的环境中,如果线程持有锁的时间非常短,自旋锁可以提高性能。

Java在ReentrantLock的实现中结合了自旋锁的概念,当锁的竞争不激烈时,避免线程进入阻塞状态。

4.3 Lock Support

Java还提供了 LockSupport 类,用于线程的挂起和唤醒。这是实现高级并发工具(如CountDownLatchCyclicBarrier等)的基础。

5. 总结

  • synchronized:简单易用,适合基本的锁定需求,但灵活性较差。
  • ReentrantLock:提供更丰富的功能,如可重入、锁超时、公平锁等,适用于复杂的并发场景。
  • ReadWriteLock:读写分离,提高读多写少场景下的并发性能。
  • 锁优化:如偏向锁、自旋锁、Lock Support等技术提高了锁的性能,适用于特定场景。

Java的锁机制虽然功能强大,但在实际开发中,我们需要根据具体场景选择合适的锁,并合理使用,以避免锁的滥用带来的性能问题。


http://www.ppmy.cn/news/1536561.html

相关文章

蒙特卡罗方法 - 重要采样篇

序言 蒙特卡罗方法,作为一种基于随机抽样的数值计算方法,在金融、物理、工程等多个领域展现出了强大的应用潜力。然而,传统蒙特卡罗方法在处理某些特定问题时,可能会遇到收敛速度慢、计算成本高等挑战。为了克服这些难题&#xf…

如何使用GeoIP和ELK(Elasticsearch、Logstash和Kibana)映射用户位置

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。 简介 IP 地理定位是用于确定 IP 地址的物理位置的过程,可以用于各种目的,如内容个性化和流量分析。通过地理位置…

利用redis实现分布式定时任务

如果是微服务,两个服务都有定时,那么就出问题了,但是上分布式定时任务框架太麻烦怎么办,那么就用redis加个锁,谁先抢到锁谁执行 整个工具类 import org.springframework.beans.factory.annotation.Autowired; import …

宠物咖啡馆数字化解决方案:基于SpringBoot的实现

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统,它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等,非常…

手机竖屏 Premiere Pro 电影转场特效视频模板Pr工程文件

10个不同的类别和115个过渡。过渡很容易使用。随附视频教程。 下载地址:Pr模板网 下载链接:https://prmuban.com/40597.html

Chromium 搜索引擎功能浅析c++

地址栏输入:chrome://settings/searchEngines 可以看到 有百度等数据源,那么如何调整其顺序呢,此数据又存储在哪里呢? 1、浏览器初始化搜索引擎数据来源在 components\search_engines\prepopulated_engines.json // Copyright …

js基础速成13-Console 对象方法

Console 对象方法 在本节中,我们将介绍 console 及其对象方法。对于绝对初学者来说,通常不知道该使用 console.log()、document.write() 还是 document.getElementById()。 我们使用 console 对象的方法在浏览器控制台显示输出,使用 docume…

目标检测or实例分割中AP、MAP的计算

参考链接: 目标检测中AP、MAP的计算_51CTO博客_目标检测map计算 举个例子: