java中锁的详解

简介: java中锁的详解

锁的种类

Java中的锁是一种同步机制,用于保护共享资源,防止多个线程同时访问。在Java中,锁可以分为不同的类型,每种类型都有其特定的优缺点和适用场景。

1.重量级锁(Heavy Lock):重量级锁是一种互斥锁,它提供了最高的线程安全性。当一个线程获得了重量级锁时,其他线程必须等待该线程释放锁才能获得锁。因此,重量级锁的开销较大,可能会导致线程阻塞和性能下降。但是,由于重量级锁提供了最高的线程安全性,所以在需要确保数据完整性和一致性的情况下,重量级锁是非常有用的。

2.轻量级锁(Lightweight Lock):轻量级锁是一种比重量级锁更轻量级的同步机制。它使用CAS操作来实现互斥访问,并且可以在没有竞争的情况下快速获取锁。因此,轻量级锁的开销较小,可以提高并发性能。但是,由于轻量级锁不是互斥的,所以如果多个线程同时尝试获取锁,则可能导致数据不一致的问题。

3.互斥锁(Mutex Lock):互斥锁是一种基本的同步机制,它用于保护共享资源,防止多个线程同时访问。互斥锁提供了原子性操作,即在任何时候只能有一个线程访问共享资源。Java中的synchronized关键字就是一种互斥锁。

4.读写锁(Read-Write Lock):读写锁允许多个线程同时读取共享资源,但只允许一个线程写入共享资源。读写锁可以提高并发性能,特别是在读操作比写操作频繁的情况下。Java中的ReentrantReadWriteLock就是一种读写锁。

5.可重入锁(Reentrant Lock):可重入锁允许同一个线程多次获取同一个锁,这意味着线程可以在执行完同步代码后再次获取该锁而不需要重新竞争。可重入锁通常用于递归调用等情况。Java中的ReentrantLock就是一种可重入锁。

6.自旋锁(Spin Lock):自旋锁是一种基于忙等待的同步机制,它在获取锁时会一直循环等待,直到获取到锁为止。自旋锁的开销较大,因为它会占用大量的CPU时间。但是,自旋锁可以提供极高的并发性能,特别是在竞争激烈的情况下。

7.乐观锁(Optimistic Lock):乐观锁是一种不涉及实际锁定的同步机制,它通过版本号或哈希值来检查共享资源是否被其他线程修改过。如果没有修改,则可以更新共享资源。乐观锁可以提高并发性能,但可能会导致数据不一致的问题。Java中的CAS操作就是一种乐观锁实现方式。

8.悲观锁(Pessimistic Lock):悲观锁是一种实际锁定的同步机制,它在访问共享资源之前先将其锁定。悲观锁可以保证数据的一致性,但会导致并发性能下降。Java中的synchronized关键字就是一种悲观锁实现方式。

以下是Java中各种锁的实现代码示例:

1. synchronized关键字实现互斥锁[悲观锁]:
public class SynchronizedDemo { 
    private int count = 0; 
    public synchronized void increment() { 
        count++; 
    } 
    public int getCount() { 
        return count; 
    } 
} 
2. ReentrantLock类实现可重入锁:
import java.util.concurrent.locks.ReentrantLock; 
public class ReentrantLockDemo { 
    private final ReentrantLock lock = new ReentrantLock(); 
    public void doSomething() { 
        lock.lock(); // 可重入锁必须先获取到锁才能执行同步代码,因此需要使用lock()方法获取锁 
        try { 
            // 同步代码块 
        } finally { 
            lock.unlock(); // 在finally块中释放锁,即使发生异常也需要释放锁 
        } 
    } 
} 
3. ReadWriteLock类实现读写锁:
import java.util.concurrent.locks.ReadWriteLock; 
import java.util.concurrent.locks.ReentrantReadWriteLock; 
public class ReadWriteLockDemo { 
    private final ReentrantReadWriteLock rwl = new ReentrantReadWriteLock(); 
    public void readData() { 
        rwl.readLock().lock(); // 只允许读取操作获取到读锁,因此需要使用readLock()方法获取读锁 
        try { 
            // 读取共享资源的操作 
        } finally { 
            rwl.readLock().unlock(); // 在finally块中释放读锁,即使发生异常也需要释放读锁 
        } 
    } 
    public void writeData() { 
        rwl.writeLock().lock(); // 只允许写入操作获取到写锁,因此需要使用writeLock()方法获取写锁 
        try { 
            // 写入共享资源的操作 
        } finally { 
            rwl.writeLock().unlock(); // 在finally块中释放写锁,即使发生异常也需要释放写锁 
        } 
    } 
} 
4.自旋锁示例代码,它使用Java中的ReentrantLock类实现:
import java.util.concurrent.locks.ReentrantLock;
public class SpinLockDemo {
    private final ReentrantLock lock = new ReentrantLock();
    public void doSomething() {
        while (true) {
            try {
                lock.lockInterruptibly(); // 尝试获取锁,如果获取失败则进入循环等待
                break;
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt(); // 抛出异常时中断当前线程
            }
        }
        try {
            // 同步代码块
        } finally {
            lock.unlock(); // 在finally块中释放锁,即使发生异常也需要释放锁
        }
    }
}

在上面的示例代码中,doSomething()方法会一直循环等待获取锁。当它尝试获取锁时,如果获取失败,则会抛出InterruptedException异常并中断当前线程。一旦它成功获取到锁,就可以执行同步代码块了。最后,无论同步代码块是否正常执行,doSomething()方法都会在finally块中释放锁。这样可以确保即使发生异常,也不会导致锁没有被正确释放的问题。

5. 乐观锁示例代码,它使用Java中的CAS操作实现:
import java.util.concurrent.atomic.AtomicInteger;
public class OptimisticLockDemo {
    private final AtomicInteger count = new AtomicInteger(0);
    public void increment() {
        int currentCount = count.get();
        while (true) {
            int nextCount = currentCount + 1;
            if (count.compareAndSet(currentCount, nextCount)) {
                break;
            } else {
                currentCount = nextCount;
            }
        }
    }
}

在上面的示例代码中,increment()方法会不断尝试将共享资源的计数器从当前值加1。如果成功更新了计数器的值,则说明没有其他线程在同一时间点对共享资源进行了修改,因此可以安全地进行同步操作。否则,increment()方法会继续循环等待,直到获取到锁为止。这样可以确保即使发生竞争条件,也不会导致数据的不一致性问题。


相关文章
|
4天前
|
安全 Java 调度
Java编程时多线程操作单核服务器可以不加锁吗?
Java编程时多线程操作单核服务器可以不加锁吗?
17 2
|
2月前
|
存储 Oracle 安全
揭秘Java并发核心:深入Hotspot源码腹地,彻底剖析Synchronized关键字的锁机制与实现奥秘!
【8月更文挑战第4天】在Java并发世界里,`Synchronized`如同导航明灯,确保多线程环境下的代码安全执行。它通过修饰方法或代码块实现独占访问。在Hotspot JVM中,`Synchronized`依靠对象监视器(Object Monitor)机制实现,利用对象头的Mark Word管理锁状态。
42 1
|
19天前
|
存储 缓存 安全
【Java面试题汇总】多线程、JUC、锁篇(2023版)
线程和进程的区别、CAS的ABA问题、AQS、哪些地方使用了CAS、怎么保证线程安全、线程同步方式、synchronized的用法及原理、Lock、volatile、线程的六个状态、ThreadLocal、线程通信方式、创建方式、两种创建线程池的方法、线程池设置合适的线程数、线程安全的集合?ConcurrentHashMap、JUC
【Java面试题汇总】多线程、JUC、锁篇(2023版)
|
7天前
|
算法 Java 关系型数据库
Java中到底有哪些锁
【9月更文挑战第24天】在Java中,锁主要分为乐观锁与悲观锁、自旋锁与自适应自旋锁、公平锁与非公平锁、可重入锁以及独享锁与共享锁。乐观锁适用于读多写少场景,通过版本号或CAS算法实现;悲观锁适用于写多读少场景,通过加锁保证数据一致性。自旋锁与自适应自旋锁通过循环等待减少线程挂起和恢复的开销,适用于锁持有时间短的场景。公平锁按请求顺序获取锁,适合等待敏感场景;非公平锁性能更高,适合频繁加解锁场景。可重入锁支持同一线程多次获取,避免死锁;独享锁与共享锁分别用于独占和并发读场景。
|
15天前
|
Java 数据库
JAVA并发编程-一文看懂全部锁机制
曾几何时,面试官问:java都有哪些锁?小白,一脸无辜:用过的有synchronized,其他不清楚。面试官:回去等通知! 今天我们庖丁解牛说说,各种锁有什么区别、什么场景可以用,通俗直白的分析,让小白再也不怕面试官八股文拷打。
|
15天前
|
安全 Java 开发者
Java并发编程中的锁机制解析
本文深入探讨了Java中用于管理多线程同步的关键工具——锁机制。通过分析synchronized关键字和ReentrantLock类等核心概念,揭示了它们在构建线程安全应用中的重要性。同时,文章还讨论了锁机制的高级特性,如公平性、类锁和对象锁的区别,以及锁的优化技术如锁粗化和锁消除。此外,指出了在高并发环境下锁竞争可能导致的问题,并提出了减少锁持有时间和使用无锁编程等策略来优化性能的建议。最后,强调了理解和正确使用Java锁机制对于开发高效、可靠并发应用程序的重要性。
16 3
|
2月前
|
存储 Java
Java锁是什么?简单了解
在高并发环境下,锁是Java中至关重要的概念。锁或互斥是一种同步机制,用于限制多线程环境下的资源访问,确保排他性和并发控制。例如,超市储物柜仅能存放一个物品,若三人同时使用,则需通过锁机制确保每次只有一个线程访问。Java中可以通过`synchronized`关键字实现加锁,确保关键代码段的原子性,避免数据不一致问题。正确使用锁可有效提升程序的稳定性和安全性。
Java锁是什么?简单了解
|
2月前
|
小程序 Java 开发工具
【Java】@Transactional事务套着ReentrantLock锁,锁竟然失效超卖了
本文通过一个生动的例子,探讨了Java中加锁仍可能出现超卖问题的原因及解决方案。作者“JavaDog程序狗”通过模拟空调租赁场景,详细解析了超卖现象及其背后的多线程并发问题。文章介绍了四种解决超卖的方法:乐观锁、悲观锁、分布式锁以及代码级锁,并重点讨论了ReentrantLock的使用。此外,还分析了事务套锁失效的原因及解决办法,强调了事务边界的重要性。
58 2
【Java】@Transactional事务套着ReentrantLock锁,锁竟然失效超卖了
|
27天前
|
Oracle Java 关系型数据库
【颠覆性升级】JDK 22:超级构造器与区域锁,重塑Java编程的两大基石!
【9月更文挑战第6天】JDK 22的发布标志着Java编程语言在性能和灵活性方面迈出了重要的一步。超级构造器和区域锁这两大基石的引入,不仅简化了代码设计,提高了开发效率,还优化了垃圾收集器的性能,降低了应用延迟。这些改进不仅展示了Oracle在Java生态系统中的持续改进和创新精神,也为广大Java开发者提供了更多的可能性和便利。我们有理由相信,在未来的Java编程中,这些新特性将发挥越来越重要的作用,推动Java技术不断向前发展。
|
2月前
|
Java 开发者
Java多线程教程:使用ReentrantLock实现高级锁功能
Java多线程教程:使用ReentrantLock实现高级锁功能
34 1
下一篇
无影云桌面