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

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

Java并發編程如何降低鎖粒度并實現性能優化

瀏覽:4日期:2022-08-25 18:09:06

在高負載多線程應用中性能是非常重要的。為了達到更好的性能,開發者必須意識到并發的重要性。當我們需要使用并發時, 常常有一個資源必須被兩個或多個線程共享。

在這種情況下,就存在一個競爭條件,也就是其中一個線程可以得到鎖(鎖與特定資源綁定),其他想要得到鎖的線程會被阻塞。這個同步機制的實現是有代價的,為了向你提供一個好用的同步模型,JVM和操作系統都要消耗資源。有三個最重要的因素使并發的實現會消耗大量資源,它們是:

上下文切換 內存同步 阻塞

為了寫出針對同步的優化代碼,你必須認識到這三個因素以及如何減少它們。在寫這樣的代碼時你需要注意很多東西。在本文中,我會向你介紹一種通過降低鎖粒度的技術來減少這些因素。讓我們從一個基本原則開始:不要長時間持有不必要的鎖。

在獲得鎖之前做完所有需要做的事,只把鎖用在需要同步的資源上,用完之后立即釋放它。我們來看一個簡單的例子:

public class HelloSync { private Map dictionary = new HashMap(); public synchronized void borringDeveloper(String key, String value) { long startTime = (new java.util.Date()).getTime(); value = value + '_'+startTime; dictionary.put(key, value); System.out.println('I did this in '+ ((new java.util.Date()).getTime() - startTime)+' miliseconds'); }}

在這個例子中,我們違反了基本原則,因為我們創建了兩個Date對象,調用了System.out.println(),還做了很多次String連接操作,但唯一需要做同步的操作是“dictionary.put(key, value);”。讓我們來修改代碼,把同步方法變成只包含這句的同步塊,得到下面更優化的代碼:

public class HelloSync { private Map dictionary = new HashMap(); public void borringDeveloper(String key, String value) { long startTime = (new java.util.Date()).getTime(); value = value + '_'+startTime; synchronized (dictionary) { dictionary.put(key, value); } System.out.println('I did this in '+ ((new java.util.Date()).getTime() - startTime)+' miliseconds'); }}

上面的代碼可以進一步優化,但這里只想傳達出這種想法。如果你對如何進一步優化感興趣,請參考java.util.concurrent.ConcurrentHashMap.

那么,我們怎么降低鎖粒度呢?簡單來說,就是通過盡可能少的請求鎖。基本的想法是,分別用不同的鎖來保護同一個類中多個獨立的狀態變量,而不是對整個類域只使用一個鎖。我們來看下面這個我在很多應用中見到過的簡單例子:

public class Grocery { private final ArrayList fruits = new ArrayList(); private final ArrayList vegetables = new ArrayList(); public synchronized void addFruit(int index, String fruit) { fruits.add(index, fruit); } public synchronized void removeFruit(int index) { fruits.remove(index); } public synchronized void addVegetable(int index, String vegetable) { vegetables.add(index, vegetable); } public synchronized void removeVegetable(int index) { vegetables.remove(index); }}

雜貨店主可以對他的雜貨鋪中的蔬菜和水果進行添加/刪除操作。上面對雜貨鋪的實現,通過基本的Grocery 鎖來保護fruits和vegetables,因為同步是在方法域完成的。事實上,我們可以不使用這個大范圍的鎖,而是針對每個資源(fruits和vegetables)分別使用一個鎖。來看一下改進后的代碼:

public class Grocery { private final ArrayList fruits = new ArrayList(); private final ArrayList vegetables = new ArrayList(); public void addFruit(int index, String fruit) { synchronized(fruits) fruits.add(index, fruit); } public void removeFruit(int index) { synchronized(fruits) {fruits.remove(index);} } public void addVegetable(int index, String vegetable) { synchronized(vegetables) vegetables.add(index, vegetable); } public void removeVegetable(int index) { synchronized(vegetables) vegetables.remove(index); }}

在使用了兩個鎖后(把鎖分離),我們會發現比起之前用一個整體鎖,鎖阻塞的情況更少了。當我們把這個技術用在有中度鎖爭搶的鎖上時,優化提升會更明顯。如果把該方法應用到輕微鎖爭搶的鎖上,改進雖然比較小,但還是有效果的。但是如果把它用在有重度鎖爭搶的鎖上時,你必須認識到結果并非總是更好。

請有選擇性的使用這個技術。如果你懷疑一個鎖是重度爭搶鎖請按下面的方法來確認是否使用上面的技術:

確認你的產品會有多少爭搶度,將這個爭搶度乘以三倍或五倍(甚至10倍,如果你想準備的萬無一失) 基于這個爭搶度做適當的測試 比較兩種方案的測試結果,然后挑選出最合適的.

用于改進同步性能的技術還有很多,但對所有的技術來說最基本的原則只有一個:不要長時間持有不必要的鎖。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Java
相關文章:
主站蜘蛛池模板: 亚洲国产亚洲片在线观看播放 | 北条麻妃在线一区二区 | 国产精品亚洲综合天堂夜夜 | 中文字幕日韩欧美一区二区三区 | 99热在线获取最新地址 | 暖暖在线精品日本中文 | 男人天堂手机在线 | 日韩a毛片免费全部播放完整 | 成人a毛片手机免费播放 | 一级特黄性色生活片一区二区 | 久久福利青草精品资源 | 92看片淫黄大片看国产片 | 国产一级片免费观看 | 久草在在线视频 | 欧美成人午夜毛片免费影院 | 国产一级高清视频 | 亚洲午夜成激人情在线影院 | 亚洲在线国产 | 99热久久国产综合精品久久国产 | 老外一级毛片免费看 | 色网站在线 | 国产成人精品视频一区 | 久久依人 | 成人影院免费看 | 国产成人不卡亚洲精品91 | 国产免费一级高清淫曰本片 | 国产九九视频在线观看 | 欧美日本色 | 国产孕妇孕交视频 | 日本一级毛片在线看 | 亚洲欧美日韩国产综合 | 亚洲视频精品 | 日本阿v视频在线观看高清 日本波多野结衣视频 | 手机看片久久高清国产日韩 | 永久免费毛片手机版在线看 | 亚洲精品色综合色在线观看 | 性欧美巨大的视频 | 精品国产一区二区三区www | 国产成人精品免费视 | 91热在线观看精品 | a级黄色毛片免费播放视频 a级精品九九九大片免费看 |