真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

java項(xiàng)目中如何利用高并發(fā)將信息寫入數(shù)據(jù)庫

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

成都網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)建站!專注于網(wǎng)頁設(shè)計(jì)、網(wǎng)站建設(shè)公司、微信開發(fā)、微信小程序、集團(tuán)成都定制網(wǎng)頁設(shè)計(jì)等服務(wù)項(xiàng)目。核心團(tuán)隊(duì)均擁有互聯(lián)網(wǎng)行業(yè)多年經(jīng)驗(yàn),服務(wù)眾多知名企業(yè)客戶;涵蓋的客戶類型包括:PVC花箱等眾多領(lǐng)域,積累了大量豐富的經(jīng)驗(yàn),同時(shí)也獲得了客戶的一致贊美!

假定存在這樣一種情況

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

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

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

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

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

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

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

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

那咋辦?

方法一

萬年不變的synchronized。

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

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

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

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

方法二

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

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

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

方法三

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

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

那咋辦?

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

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

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

有可能張三李四的后綴一樣,但是張三李四同時(shí)發(fā)請求的概率,應(yīng)該也不會太大。就算真的同時(shí)發(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   
  }   
} 
 

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

方法五

如果是集群情況下,兩個張三幾乎瞬時(shí)進(jìn)入兩臺服務(wù)器,那java語言級別的鎖都得報(bào)廢。

可以使用redis的分布式鎖

方法六

使用zookeeper

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

以上就是java項(xiàng)目中如何利用高并發(fā)將信息寫入數(shù)據(jù)庫,小編相信有部分知識點(diǎn)可能是我們?nèi)粘9ぷ鲿姷交蛴玫降?。希望你能通過這篇文章學(xué)到更多知識。更多詳情敬請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


分享名稱:java項(xiàng)目中如何利用高并發(fā)將信息寫入數(shù)據(jù)庫
轉(zhuǎn)載來于:http://weahome.cn/article/joidgg.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部