欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

synchronized关键字之死锁的原因

程序员文章站 2022-05-04 10:09:58
...

1、交叉锁可导致程序出现死锁

线程A持有R1的锁等待获取R2的锁,线程B持有R2的锁等待获取R1的锁(典型的哲学家吃面),这种情况最容易导致程序发生死锁的问题,示例如下:

package com.practice.concurrent.chapter04.sync;

public class DeadLock {
    private final Object MUTEX_READ=new Object();
    private final Object MUTEX_WRITE=new Object();
    public void read(){
        synchronized (MUTEX_READ){
            System.out.println(Thread.currentThread().getName()+" get read lock");
            synchronized (MUTEX_WRITE){
                System.out.println(Thread.currentThread().getName()+" get write lock");
            }
            System.out.println(Thread.currentThread().getName()+" release write lock");
        }
        System.out.println(Thread.currentThread().getName()+" release read lock");
    }
    public void write(){
        synchronized (MUTEX_WRITE){
            System.out.println(Thread.currentThread().getName()+" get write lock");
            synchronized (MUTEX_READ){
                System.out.println(Thread.currentThread().getName()+" get read lock");
            }
            System.out.println(Thread.currentThread().getName()+" release read lock");
        }
        System.out.println(Thread.currentThread().getName()+" release write lock");
    }

    public static void main(String[] args) {
        final DeadLock deadLock = new DeadLock();
        new Thread(()->{
            while (true){
                deadLock.read();
            }
        },"thread-read").start();
        new Thread(()->{
            while (true){
                deadLock.write();
            }
        },"thread-write").start();
    }
}

运行结果截图:

synchronized关键字之死锁的原因

2、内存不足

当并发请求系统可用内存时,如果此时系统内存不足,则可能会出现死锁的情况。举个例子,两个线程T1和T2,执行某个任务,其中T1已经获取了10MB内存,T2获取了20MB内存,如果每个线程的执行单元都需要30MB内存,但是剩余可用的内存没有了,那么两个线程有可能都在等待彼此能够释放内存资源。

3、一问一答式的数据交换

服务端开启某个端口,等待客户端访问 ,客户端发送请求立即等待接收,由于某种原因服务端错过了客户端的请求,仍然在等待一问一答式的数据交换,此时服务端和客户端都在等待着双方发送数据。

4、数据库锁

无论是数据库表级别的锁,还是行级别的锁,比如某个线程执行for update语句退出了事务,其他线程访问该数据库时都将陷入死锁

5、文件锁

同理,某线程获得了文件锁意外退出,其他读取该文件的线程也将会进入死锁知道系统释放文件句柄资源。

6、死循环引起的死锁

程序由于代码原因或者对某些异常处理的不得当,进入了死循环,虽然查看线程堆栈信息不回发现任何死锁的迹象,但是程序不工作,cpu占有率又居高不下,这种死锁一般称为系统假死,是一种最为致命也是最难排查的死锁现象,由于重现困难,进程对系统资源的使用量又达到了极限,想要做出dump有时候也是非常困难的。

注意:hashmap不是线程安全的,如果在多线程中使用并发访问可能会导致死锁,多线程中并发访问建议使用ConcurrentHashMap

 

相关标签: Java高并发编程