自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java并發(fā):如何避免死鎖

開發(fā) 前端
一般出現(xiàn)死鎖時,可能會導致CPU、內(nèi)存等資源消耗過高,導致系統(tǒng)性能下降。也可能導致應用無響應或者假死等等,所以要從多角度進行死鎖的排查。

一般在Java項目里用到鎖的場景不多,有朋友調(diào)侃說用到鎖的次數(shù)還沒有面試被問到的次數(shù)多,哈哈!

1.死鎖如何產(chǎn)生

說句難聽話,鎖一般都很少用到,何況死鎖呢?想產(chǎn)生死鎖還是有點難的,需要滿足2個條件:

共享資源同時只能被一個線程使用,如果已經(jīng)有一個線程占用了資源,其余線程只能等待,直到資源被釋放。

死鎖情況肯定存在多個資源被多個線程爭搶的情況。

比如線程1持有了資源A,然后去等待獲取資源B,線程2持有了資源B,然后等待獲取資源A,這樣就會形成死鎖。

2.如何避免死鎖

一般有2種方式避免死鎖:

  • 線程一次性獲取需要的全部資源。
  • 獲取鎖時,增加超時動作。如果在一定的時間內(nèi)獲取不到鎖,則釋放已經(jīng)獲取的鎖。

3.代碼實踐

/**
 * 避免死鎖,我覺得有2種方式:
 * 1、線程直接一把頭獲取所需要的全部鎖,不要分步
 * 2、線程獲取A之后,再去獲取B,超時仍未獲取到B,則釋放A
 */
public class AvoidDeadLock01 {

    private static Lock lock1 = new ReentrantLock();
    private static Lock lock2 = new ReentrantLock();

    public static void acquireLocks(Lock lock1, Lock lock2) {
        boolean isLock1Acquired = false;
        boolean isLock2Acquired = false;
        while (true) {
            try {
                isLock1Acquired = lock1.tryLock();
                isLock2Acquired = lock2.tryLock();
            } finally {
                if (isLock1Acquired && isLock2Acquired) {
                    return;
                }
                if (isLock1Acquired) {
                    lock1.unlock();
                }
                if (isLock2Acquired) {
                    lock2.unlock();
                }
            }

            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
    }

    public static void main(String[] args) {
        Thread thread1 = new Thread(() -> {
            acquireLocks(lock1, lock2);
            System.out.println("=====線程1 獲取到了2把鎖=====");
            lock1.unlock();
            lock2.unlock();
        });
        Thread thread2 = new Thread(() -> {
            acquireLocks(lock1, lock2);
            System.out.println("=====線程2 獲取到了2把鎖=====");
            lock1.unlock();
            lock2.unlock();
        });
        thread1.start();
        thread2.start();
    }

}
public class AvoidDeadLock02 {

    private static Lock lock1 = new ReentrantLock();
    private static Lock lock2 = new ReentrantLock();

    public static void acquireLocks(Lock lock1, Lock lock2) {
        boolean isLock1Acquired = false;
        boolean isLock2Acquired = false;

        try {
            while (true) {
                isLock1Acquired = lock1.tryLock(200, TimeUnit.MILLISECONDS);
                if (isLock1Acquired) {
                    isLock2Acquired = lock2.tryLock(200, TimeUnit.MILLISECONDS);
                    if (isLock2Acquired) {
                        break;
                    } else {
                        lock1.unlock();
                    }
                }
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        } finally {
            if (!isLock1Acquired || !isLock2Acquired) {
                if (isLock1Acquired) {
                    lock1.unlock();
                }
                if (isLock2Acquired) {
                    lock2.unlock();
                }
            }
        }
    }

    public static void main(String[] args) {
        Thread thread1 = new Thread(() -> {
            acquireLocks(lock1, lock2);
            System.out.println("=====線程1 獲取到了2把鎖=====");
            lock1.unlock();
            lock2.unlock();
        });
        Thread thread2 = new Thread(() -> {
            acquireLocks(lock1, lock2);
            System.out.println("=====線程2 獲取到了2把鎖=====");
            lock1.unlock();
            lock2.unlock();
        });
        thread1.start();
        thread2.start();
    }

}

4.出現(xiàn)死鎖如何排查

一般出現(xiàn)死鎖時,可能會導致CPU、內(nèi)存等資源消耗過高,導致系統(tǒng)性能下降。也可能導致應用無響應或者假死等等,所以要從多角度進行死鎖的排查。

首先是用top、df、free等命令查看操作系統(tǒng)的基本情況。然后可以使用jmap、jstack等命令查看JVM線程棧和堆內(nèi)存的情況。一般出現(xiàn)死鎖時,會在線程棧的信息里出現(xiàn)deadlock字樣。

還可以采用VisualVM、JConsole等工具進行排查。

責任編輯:趙寧寧 來源: 不焦躁程序員
相關推薦

2010-03-16 18:06:29

Java線程死鎖

2019-10-29 16:10:55

死鎖Java并發(fā)

2024-11-20 08:00:00

死鎖多線程編程

2010-06-29 17:32:13

SQL Server鎖

2017-05-03 16:26:24

MySQL并發(fā)死鎖

2023-12-08 18:01:25

Java關鍵字

2010-07-20 08:56:37

SQL Server鎖

2021-03-03 08:57:46

java死鎖線程

2015-03-18 09:21:53

編程Java死鎖

2025-03-03 04:00:00

線程安全CPU

2009-06-12 16:15:42

死鎖Java虛擬機

2023-12-29 09:28:25

Java編程

2019-11-21 08:51:56

Java異常控制流

2015-07-30 10:12:32

JavaNullAssert

2024-07-01 08:00:00

2023-07-18 09:24:04

MySQL線程

2018-05-29 11:44:22

數(shù)據(jù)庫MySQL死鎖

2024-08-19 11:45:18

2024-09-29 00:00:00

高并發(fā)交易所宕機

2024-04-01 00:01:05

MySQL死鎖DDL
點贊
收藏

51CTO技術棧公眾號