tomcat - java數(shù)據(jù)存放問(wèn)題
問(wèn)題描述
如題,有登陸的系統(tǒng),老項(xiàng)目沒(méi)cache(我沒(méi)權(quán)利去加),但是不同的服務(wù)都要使用先前http請(qǐng)求到的數(shù)據(jù)(由用戶區(qū)分),想把它保存起來(lái)避免每次重復(fù)發(fā)http請(qǐng)求浪費(fèi)資源。
后臺(tái)springmvc
目前我想到3個(gè)方法:
1.丟session里面(HttpSessionListener),應(yīng)該最簡(jiǎn)單,但不知道潛在問(wèn)題2.丟threadlocal里面(controller搞個(gè)static 的threadlocal的變量,或者寫(xiě)個(gè)contextholder)3.controller搞個(gè)ConcurrentHashMap的成員,把數(shù)據(jù)按<用戶id,http請(qǐng)求拿到的數(shù)據(jù)>放進(jìn)去.但是這個(gè)肯定不可行,可能會(huì)導(dǎo)致堆區(qū)OOF
說(shuō)說(shuō)第2個(gè)方案可能存在的問(wèn)題。1.網(wǎng)上說(shuō)的可能內(nèi)存泄露問(wèn)題,導(dǎo)致PermGen出現(xiàn)OOF,原文連接ThreadLocal 內(nèi)存泄露的實(shí)例分析
我不確定是否會(huì)出現(xiàn)問(wèn)題(原文有點(diǎn)看不懂),因?yàn)門hreadLocalMap的set具有保護(hù)機(jī)制
2.會(huì)不會(huì)出現(xiàn)請(qǐng)求線程里面的數(shù)據(jù)串了,比如1個(gè)請(qǐng)求線程同時(shí)服務(wù)兩個(gè)用戶(A和B)請(qǐng)求,B把自己的數(shù)據(jù)放到請(qǐng)求線程,覆蓋了A的,而請(qǐng)求線程服務(wù)A的時(shí)候,拿到了B的數(shù)據(jù)。。
問(wèn)題解答
回答1:方法1是最簡(jiǎn)單、最常用的,如果用戶量太大,或者做了負(fù)載均衡,就要實(shí)現(xiàn)集中存儲(chǔ)的Session,有很多現(xiàn)成的方案可以支持集中存儲(chǔ)的HttpSession的,存Redis、MongoDB、MySQL的都有,GitHub上搜一下。
方法2不解決問(wèn)題,主要是因?yàn)橛脩舻卿浐螅啻握?qǐng)求可能會(huì)落在多個(gè)線程里。你說(shuō)的第二點(diǎn)也是理由。
方法3也是一種實(shí)現(xiàn)方式,其實(shí)Tomcat的HttpSession就是用ConcurrentHashMap實(shí)現(xiàn)的(只是它用sessionId而不是用userId做key),但要注意的一點(diǎn)是,你必須自己管理Map中每個(gè)Key-Value的生命周期,例如Session超時(shí)了要及時(shí)remove掉。
相關(guān)文章:
1. mysql - 10g數(shù)據(jù)庫(kù)如何遷移2. php - 有關(guān)sql語(yǔ)句反向LIKE的處理3. 在視圖里面寫(xiě)php原生標(biāo)簽不是要迫不得已的情況才寫(xiě)嗎4. 獲取上次登錄ip的原理是啥?5. node.js - session怎么存到cookie,然后服務(wù)器重啟后還能獲取。數(shù)據(jù)庫(kù)不用mongodb或redis,數(shù)據(jù)庫(kù)是mysql6. 求救一下,用新版的phpstudy,數(shù)據(jù)庫(kù)過(guò)段時(shí)間會(huì)消失是什么情況?7. 為什么說(shuō)非對(duì)象調(diào)用成員函數(shù)fetch()8. fetch_field_direct()報(bào)錯(cuò)9. 為什么點(diǎn)擊登陸沒(méi)反應(yīng)10. mysql多表聯(lián)合查詢優(yōu)化的問(wèn)題
