91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

java項目中如何利用高并發將信息寫入數據庫

發布時間:2020-11-11 15:46:42 來源:億速云 閱讀:467 作者:Leah 欄目:編程語言

本篇文章給大家分享的是有關java項目中如何利用高并發將信息寫入數據庫,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

假定存在這樣一種情況

多個用戶對數據庫進行寫,我們的業務邏輯規定,每個用戶只能寫一次,大部分用戶也只發一次請求。

public void write(Uers u){ 
 // do something 
} 

但是有一種情況(1%的情況下吧)的就是有的用戶會發兩次甚至更多次寫請求(因為數據庫限制,我們不方便在主鍵上做文章)。

如果這個特殊的用戶發送的兩次請求時間間隔比較大,那就簡單了,再每次寫入的時候,寫去數據庫里看看,這個人有沒有寫過,如果已經寫過了,就直接拋棄這個請求。

public void write(Uers u){ 
 if(!checkIfExistUser(u)){ 
   // do something 
  } 
} 

不過最大的問題就是,如果用戶幾乎在瞬時,發送了兩個寫操作。

而且假定我們的do something比較耗時,那么上面的策略就有可能失敗。

為啥失敗?我不用解釋了吧。

那咋辦?

方法一

萬年不變的synchronized。

public synchronized void write(Uers u){ 
 if(!checkIfExistUser(u)){ 
   // do something 
  } 
} 

當然,我們得承認,有了上面的方法,就不會出現,數據庫里有兩條張三的記錄了

但上面的鎖的粒度太大了,張三寫的時候,李四也不能寫了。

其實我們想要的只是:張三自己本人,不能同時多次寫入。

方法二

類 String 維護一個字符串池。 當調用 intern 方法時,如果池已經包含一個等于此 String 對象的字符串(該對象由 equals(Object) 方法確定),則返回池中的字符串。可見,當String相同時,String.intern()總是返回同一個對象,因此就實現了對同一用戶加鎖。由于鎖的粒度局限于具體用戶,使系統獲得了最大程度的并發。

public synchronized void write(Uers u){ 
  synchronized(u.getUserId.intern()) { 
   // do something 
  } 
}

上面的思路就保證了張三寫的時候,李四可以寫,但是不能兩個張三一塊寫。

方法三

其實我個人覺得,方法二已經很好了,如果非要說方法二還有什么問題的話,只能說:

String.inter()的缺陷是類 String 維護一個字符串池是放在JVM perm區的,如果用戶數特別多,導致放入字符串池的String不可控,有可能導致OOM錯誤或者過多的Full GC。

那咋辦?

public synchronized void write(Uers u){ 
  String userSuffix=getSuffix(u); 
  synchronized(userSuffix.intern()) { 
   // do something 
  } 
} 

至于那個獲得后綴的策略,大家自己想。

有了這個策略,我就能保證1億個用戶,可能只有10000個不同的后綴。

有可能張三李四的后綴一樣,但是張三李四同時發請求的概率,應該也不會太大。就算真的同時發了,那你等一下不行么?

方法四

Map locks = new Map();   
List lockKeys = new List();   
for(int number : 1 - 10000) {   
  Object lockKey = new Object();   
  lockKeys.add(lockKey);   
  locks.put(lockKey, new Object());   
}    
public void doSomeThing(String uid) {   
  Object lockKey = lockKeys.get(uid.hash() % lockKeys.size());   
  Object lock = locks.get(lockKey);   
  synchronized(lock) {   
   // do something   
  }   
} 
 

個人感覺和方法三的核心差不多。

方法五

如果是集群情況下,兩個張三幾乎瞬時進入兩臺服務器,那java語言級別的鎖都得報廢。

可以使用redis的分布式鎖

方法六

使用zookeeper

只是聽說有這么一個思路,但是本人沒用過zookeeper,這個方法就不多說了。

以上就是java項目中如何利用高并發將信息寫入數據庫,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

明溪县| 岗巴县| 伊通| 泉州市| 金昌市| 平昌县| 肃北| 壤塘县| 大竹县| 郧西县| 墨玉县| 炎陵县| 积石山| 博白县| 广平县| 电白县| 册亨县| 井陉县| 岑溪市| 舟曲县| 蒙城县| 彭山县| 福泉市| 湟中县| 斗六市| 山丹县| 惠东县| 公安县| 鄂托克前旗| 朔州市| 大同县| 台南市| 张家界市| 贵阳市| 永春县| 屏东市| 潮州市| 西城区| 莒南县| 疏附县| 阿克|