欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java高并發(fā)寫入用戶信息到數(shù)據(jù)庫的幾種方法

 更新時間:2017年03月02日 16:24:28   作者:程序員小董  
本文主要介紹了java高并發(fā)寫入用戶信息到數(shù)據(jù)庫的幾種方法,具有很好的參考價值。下面跟著小編一起來看下吧

假定存在這樣一種情況

多個用戶對數(shù)據(jù)庫進行寫,我們的業(yè)務邏輯規(guī)定,每個用戶只能寫一次,大部分用戶也只發(fā)一次請求。

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

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

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

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

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

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

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

那咋辦?

方法一

萬年不變的synchronized。

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

當然,我們得承認,有了上面的方法,就不會出現(xiàn),數(shù)據(jù)庫里有兩條張三的記錄了

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

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

方法二

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

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

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

方法三

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

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

那咋辦?

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

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

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

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

方法四

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,這個方法就不多說了。

以上就是本文的全部內容,希望本文的內容對大家的學習或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!

相關文章

  • RabbitMQ延時隊列詳解與Java代碼實現(xiàn)

    RabbitMQ延時隊列詳解與Java代碼實現(xiàn)

    這篇文章主要介紹了RabbitMQ延時隊列詳解與Java代碼實現(xiàn),RabbitMQ 延時隊列是指消息在發(fā)送到隊列后,并不立即被消費者消費,而是等待一段時間后再被消費者消費。這種隊列通常用于實現(xiàn)定時任務,需要的朋友可以參考下
    2023-04-04
  • Java對象的內存布局全流程

    Java對象的內存布局全流程

    這篇文章主要介紹了Java對象的內存布局全流程,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • mybatis實現(xiàn)增刪改查_動力節(jié)點Java學院整理

    mybatis實現(xiàn)增刪改查_動力節(jié)點Java學院整理

    本文通過實例代碼給大家介紹了mybatis實現(xiàn)增刪改查功能,非常不錯,具有參考借鑒價值,需要的朋友參考下吧
    2017-09-09
  • Springboot整合Gson報錯問題解決過程

    Springboot整合Gson報錯問題解決過程

    這篇文章主要介紹了Springboot整合Gson報錯問題解決過程,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-06-06
  • Java實現(xiàn)導出ZIP壓縮包的方法

    Java實現(xiàn)導出ZIP壓縮包的方法

    這篇文章主要介紹了Java實現(xiàn)導出ZIP壓縮包的方法,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-11-11
  • 解決Mybatis-plus找不到對應表及默認表名命名規(guī)則的問題

    解決Mybatis-plus找不到對應表及默認表名命名規(guī)則的問題

    這篇文章主要介紹了解決Mybatis-plus找不到對應表及默認表名命名規(guī)則的問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-11-11
  • 淺析java的foreach循環(huán)

    淺析java的foreach循環(huán)

    foreach語句是java5之后的新特征之一,在循環(huán)遍歷數(shù)組、集合方面更加簡潔,有需要的朋友可以參考一下
    2013-12-12
  • Java PDF 添加數(shù)字簽名的實現(xiàn)方法

    Java PDF 添加數(shù)字簽名的實現(xiàn)方法

    這篇文章主要介紹了Java PDF 添加數(shù)字簽名的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-12-12
  • Spring boot + thymeleaf 后端直接給onclick函數(shù)賦值的實現(xiàn)代碼

    Spring boot + thymeleaf 后端直接給onclick函數(shù)賦值的實現(xiàn)代碼

    這篇文章主要介紹了Spring boot + thymeleaf 后端直接給onclick函數(shù)賦值的實現(xiàn)代碼,需要的朋友可以參考下
    2017-06-06
  • Java實現(xiàn)經(jīng)典游戲2048的示例代碼

    Java實現(xiàn)經(jīng)典游戲2048的示例代碼

    2014年Gabriele Cirulli利用周末的時間寫2048這個游戲的程序。本文將用java語言實現(xiàn)這一經(jīng)典游戲,并采用了swing技術進行了界面化處理,需要的可以參考一下
    2022-02-02

最新評論