国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術文章
文章詳情頁

淺談java安全編碼指南之死鎖dead lock

瀏覽:2日期:2022-08-11 11:47:51
目錄不同的加鎖順序使用private類變量使用相同的Order釋放掉已占有的鎖不同的加鎖順序

我們來看一個不同加鎖順序的例子:

public class DiffLockOrder { private int amount; public DiffLockOrder(int amount){ this.amount=amount; } public void transfer(DiffLockOrder target,int transferAmount){synchronized (this){ synchronized (target){if(amount< transferAmount){ System.out.println('余額不足!');}else{ amount=amount-transferAmount; target.amount=target.amount+transferAmount;} }} }}

上面的例子中,我們模擬一個轉賬的過程,amount用來表示用戶余額。transfer用來將當前賬號的一部分金額轉移到目標對象中。

為了保證在transfer的過程中,兩個賬戶不被別人修改,我們使用了兩個synchronized關鍵字,分別把transfer對象和目標對象進行鎖定。

看起來好像沒問題,但是我們沒有考慮在調用的過程中,transfer的順序是可以發送變化的:

DiffLockOrder account1 = new DiffLockOrder(1000);DiffLockOrder account2 = new DiffLockOrder(500);Runnable target1= ()->account1.transfer(account2,200);Runnable target2= ()->account2.transfer(account1,100);new Thread(target1).start();new Thread(target2).start();

上面的例子中,我們定義了兩個account,然后兩個賬戶互相轉賬,最后很有可能導致互相鎖定,最后產生死鎖。

使用private類變量

使用兩個sync會有順序的問題,那么有沒有辦法只是用一個sync就可以在所有的實例中同步呢?

有的,我們可以使用private的類變量,因為類變量是在所有實例中共享的,這樣一次sync就夠了:

public class LockWithPrivateStatic { private int amount; private static final Object lock = new Object(); public LockWithPrivateStatic(int amount){ this.amount=amount; } public void transfer(LockWithPrivateStatic target, int transferAmount){synchronized (lock) { if (amount < transferAmount) {System.out.println('余額不足!'); } else {amount = amount - transferAmount;target.amount = target.amount + transferAmount; }} }}使用相同的Order

我們產生死鎖的原因是無法控制上鎖的順序,如果我們能夠控制上鎖的順序,是不是就不會產生死鎖了呢?

帶著這個思路,我們給對象再加上一個id字段:

private final long id; // 唯一ID,用來排序private static final AtomicLong nextID = new AtomicLong(0); // 用來生成IDpublic DiffLockWithOrder(int amount){ this.amount=amount; this.id = nextID.getAndIncrement();}

在初始化對象的時候,我們使用static的AtomicLong類來為每個對象生成唯一的ID。

在做transfer的時候,我們先比較兩個對象的ID大小,然后根據ID進行排序,最后安裝順序進行加鎖。這樣就能夠保證順序,從而避免死鎖。

public void transfer(DiffLockWithOrder target, int transferAmount){ DiffLockWithOrder fist, second; if (compareTo(target) < 0) {fist = this;second = target; } else {fist = target;second = this; } synchronized (fist){synchronized (second){ if(amount< transferAmount){System.out.println('余額不足!'); }else{amount=amount-transferAmount;target.amount=target.amount+transferAmount; }} }}釋放掉已占有的鎖

死鎖是互相請求對方占用的鎖,但是對方的鎖一直沒有釋放,我們考慮一下,如果獲取不到鎖的時候,自動釋放已占用的鎖是不是也可以解決死鎖的問題呢?

因為ReentrantLock有一個tryLock()方法,我們可以使用這個方法來判斷是否能夠獲取到鎖,獲取不到就釋放已占有的鎖。

我們使用ReentrantLock來完成這個例子:

public class DiffLockWithReentrantLock { private int amount; private final Lock lock = new ReentrantLock(); public DiffLockWithReentrantLock(int amount){this.amount=amount; } private void transfer(DiffLockWithReentrantLock target, int transferAmount) throws InterruptedException {while (true) { if (this.lock.tryLock()) {try { if (target.lock.tryLock()) {try { if(amount< transferAmount){System.out.println('余額不足!'); }else{amount=amount-transferAmount;target.amount=target.amount+transferAmount; } break;} finally { target.lock.unlock();} }} finally { this.lock.unlock();} } //隨機sleep一定的時間,保證可以釋放掉鎖 Thread.sleep(1000+new Random(1000L).nextInt(1000));} }}

我們把兩個tryLock方法在while循環中,如果不能獲取到鎖就循環遍歷。

以上就是淺談java安全編碼指南之死鎖dead lock的詳細內容,更多關于java安全編碼指南之死鎖dead lock的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
主站蜘蛛池模板: 成人手机视频在线观看 | 国产成人综合网在线播放 | 极品丝袜高跟91白沙发在线 | 在线播放高清国语自产拍免费 | 欧美一级二级三级 | 亚洲精品在线播放视频 | 深夜做爰性大片很黄很色视频 | 成人伊人青草久久综合网 | 国产成人一区二区三区免费观看 | 亚洲精品一区二区三区网址 | 亚洲欧美日韩精品久久久 | 欧美一级片在线视频 | 一级做a爰片久久毛片美女 一级做a爰片久久毛片免费看 | 欧美大陆日韩 | 亚州人成网在线播放 | 欧美最大成人毛片视频网站 | 亚洲国产日韩欧美在线 | 欧美一级毛片免费看 | 欧美成人亚洲高清在线观看 | 久久久国产高清 | 日本加勒比在线播放 | 看欧美毛片一级毛片 | 一级黄色毛片免费看 | 亚洲欧洲一区二区 | 国内真实愉拍系列情侣 | 免费一级毛片私人影院a行 免费一级毛片无毒不卡 | 国产成人小视频 | 一级做a级爰片性色毛片视频 | 自拍偷拍欧美视频 | 成人免费一级毛片在线播放视频 | jizjiz日本 | 欧美成人亚洲综合精品欧美激情 | 天天夜夜久久 | 国产特级全黄一级毛片不卡 | 大量真实偷拍情侣视频野战 | 喷潮白浆 | 在线视频一区二区 | 国产99视频精品免费观看9e | 一级作爱视频免费观看 | 欧美成人精品不卡视频在线观看 | 亚洲国产欧洲精品路线久久 |