synchronized关键字之死锁的原因

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();
    }
}

运行结果截图:

2、内存不足

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

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

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

4、数据库锁

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

5、文件锁

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

6、死循环引起的死锁

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

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

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值