死锁是线程间争夺资源造成的无限等待现象,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

相关文章
|
7月前
|
Java 开发工具
【Azure Storage Account】Java Code访问Storage Account File Share的上传和下载代码示例
本文介绍如何使用Java通过azure-storage-file-share SDK实现Azure文件共享的上传下载。包含依赖引入、客户端创建及完整示例代码,助你快速集成Azure File Share功能。
519 6
|
7月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
396 1
|
7月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
368 1
|
8月前
|
IDE Java 关系型数据库
Java 初学者学习路线(含代码示例)
本教程为Java初学者设计,涵盖基础语法、面向对象、集合、异常处理、文件操作、多线程、JDBC、Servlet及MyBatis等内容,每阶段配核心代码示例,强调动手实践,助你循序渐进掌握Java编程。
1102 3
|
8月前
|
Java
java入门代码示例
本文介绍Java入门基础,包含Hello World、变量类型、条件判断、循环及方法定义等核心语法示例,帮助初学者快速掌握Java编程基本结构与逻辑。
652 0
|
8月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java API 开发者
301 0
Java 数据库 Spring
351 0
|
8月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
537 16

热门文章

最新文章