死锁是线程间争夺资源造成的无限等待现象,Java示例展示了两个线程各自持有资源并等待对方释放,导致死锁。`

简介: 【6月更文挑战第20天】死锁是线程间争夺资源造成的无限等待现象,Java示例展示了两个线程各自持有资源并等待对方释放,导致死锁。`volatile`保证变量的可见性和部分原子性,确保多线程环境中值的即时更新。与`synchronized`相比,`volatile`作用于单个变量,不保证原子操作,同步范围有限,但开销较小。`synchronized`提供更全面的内存语义,保证原子性和可见性,适用于复杂并发控制。

死锁:
死锁是指两个或多个线程在执行过程中,因争夺资源而造成的一种相互等待的现象。若无外力干涉,它们都将无法推进下去。这种情况通常发生在多个线程都占有部分共享资源但又都在等待其它线程释放自己需要的资源时。

例如,在Java中,考虑以下场景:

public class DeadlockExample {
   
    private static Object resource1 = new Object();
    private static Object resource2 = new Object();

    public static void main(String[] args) {
   
        Thread thread1 = new Thread(() -> {
   
            synchronized (resource1) {
   
                System.out.println("Thread 1: Acquired resource 1");
                try {
   
                    Thread.sleep(500);
                } catch (InterruptedException e) {
   
                    e.printStackTrace();
                }
                synchronized (resource2) {
   
                    System.out.println("Thread 1: Acquired resource 2");
                }
            }
        });

        Thread thread2 = new Thread(() -> {
   
            synchronized (resource2) {
   
                System.out.println("Thread 2: Acquired resource 2");
                try {
   
                    Thread.sleep(500);
                } catch (InterruptedException e) {
   
                    e.printStackTrace();
                }
                synchronized (resource1) {
   
                    System.out.println("Thread 2: Acquired resource 1");
                }
            }
        });

        thread1.start();
        thread2.start();
    }
}

在这个例子中,thread1thread2分别尝试获取对方已经持有的资源,导致两者都无法继续执行。因此,这个程序会陷入死锁状态。

Java中的volatile关键字:
volatile关键字用于修饰变量,它可以确保对变量的更新操作能被其他线程看到。当一个变量被声明为volatile后,Java内存模型保证了对该变量的读写操作都是原子性的,且每次读取该变量时都会从主内存中读取最新值,而不是从工作内存(每个线程都有自己独立的工作内存)中读取。

volatile与synchronized的区别:

  • 同步范围volatile仅能用于修饰变量,而synchronized可以作用于代码块、方法以及类。
  • 内存语义volatile仅保证了可见性,即一个线程修改了volatile变量的值后,其他线程可以立即看到该变化;而synchronized除了保证可见性,还保证了原子性,即一次只有一个线程可以访问被锁定的对象或代码块。
  • 执行顺序synchronized可以确保指令重排序不会影响到多线程之间的数据一致性,而volatile不能防止指令重排序。
  • 性能开销:由于synchronized提供了更多的保障,所以它的性能开销比volatile更大。

总的来说,volatile适合用于简单的同步需求,比如确保某个标志位的改变能够及时地被其他线程看到。而对于复杂的并发控制场景,比如需要保证数据完整性的情况,应使用synchronized或其他高级并发工具如Lock

相关文章
|
13天前
|
安全 算法 Java
Java 多线程:线程安全与同步控制的深度解析
本文介绍了 Java 多线程开发的关键技术,涵盖线程的创建与启动、线程安全问题及其解决方案,包括 synchronized 关键字、原子类和线程间通信机制。通过示例代码讲解了多线程编程中的常见问题与优化方法,帮助开发者提升程序性能与稳定性。
56 0
|
1月前
|
存储 SQL 安全
Java 无锁方式实现高性能线程实战操作指南
本文深入探讨了现代高并发Java应用中单例模式的实现方式,分析了传统单例(如DCL)的局限性,并提出了多种无锁实现方案。包括基于ThreadLocal的延迟初始化、VarHandle原子操作、Record不可变对象、响应式编程(Reactor)以及CDI依赖注入等实现方式。每种方案均附有代码示例及适用场景,同时通过JMH性能测试对比各实现的优劣。最后,结合实际案例设计了一个高性能配置中心,展示了无锁单例在实际开发中的应用。总结中提出根据场景选择合适的实现方式,并遵循现代单例设计原则以优化性能和安全性。文中还提供了代码获取链接,便于读者实践与学习。
51 0
|
11天前
|
Java API 调度
从阻塞到畅通:Java虚拟线程开启并发新纪元
从阻塞到畅通:Java虚拟线程开启并发新纪元
153 83
|
16天前
|
Arthas 监控 Java
Java死锁 如何定位?如何避免Java死锁?(图解+秒懂+史上最全)
Java死锁 如何定位?如何避免Java死锁?(图解+秒懂+史上最全)
Java死锁 如何定位?如何避免Java死锁?(图解+秒懂+史上最全)
|
17天前
|
存储 Java 调度
Java虚拟线程:轻量级并发的革命性突破
Java虚拟线程:轻量级并发的革命性突破
164 83
|
1月前
|
移动开发 Java
说一说 Java 是如何实现线程间通信
我是小假 期待与你的下一次相遇 ~
|
1月前
|
存储 Java
说一说 JAVA 内存模型与线程
我是小假 期待与你的下一次相遇 ~
|
1月前
|
Java 数据挖掘 调度
Java 多线程创建零基础入门新手指南:从零开始全面学习多线程创建方法
本文从零基础角度出发,深入浅出地讲解Java多线程的创建方式。内容涵盖继承`Thread`类、实现`Runnable`接口、使用`Callable`和`Future`接口以及线程池的创建与管理等核心知识点。通过代码示例与应用场景分析,帮助读者理解每种方式的特点及适用场景,理论结合实践,轻松掌握Java多线程编程 essentials。
105 5
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问