tomcat - java數(shù)據(jù)存放問題
問題描述
如題,有登陸的系統(tǒng),老項(xiàng)目沒cache(我沒權(quán)利去加),但是不同的服務(wù)都要使用先前http請(qǐng)求到的數(shù)據(jù)(由用戶區(qū)分),想把它保存起來避免每次重復(fù)發(fā)http請(qǐng)求浪費(fèi)資源。
后臺(tái)springmvc
目前我想到3個(gè)方法:
1.丟session里面(HttpSessionListener),應(yīng)該最簡(jiǎn)單,但不知道潛在問題2.丟threadlocal里面(controller搞個(gè)static 的threadlocal的變量,或者寫個(gè)contextholder)3.controller搞個(gè)ConcurrentHashMap的成員,把數(shù)據(jù)按<用戶id,http請(qǐng)求拿到的數(shù)據(jù)>放進(jìn)去.但是這個(gè)肯定不可行,可能會(huì)導(dǎo)致堆區(qū)OOF
說說第2個(gè)方案可能存在的問題。1.網(wǎng)上說的可能內(nèi)存泄露問題,導(dǎo)致PermGen出現(xiàn)OOF,原文連接ThreadLocal 內(nèi)存泄露的實(shí)例分析
我不確定是否會(huì)出現(xià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ù)。。
問題解答
回答1:方法1是最簡(jiǎn)單、最常用的,如果用戶量太大,或者做了負(fù)載均衡,就要實(shí)現(xiàn)集中存儲(chǔ)的Session,有很多現(xiàn)成的方案可以支持集中存儲(chǔ)的HttpSession的,存Redis、MongoDB、MySQL的都有,GitHub上搜一下。
方法2不解決問題,主要是因?yàn)橛脩舻卿浐螅啻握?qǐng)求可能會(huì)落在多個(gè)線程里。你說的第二點(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. boot2docker無法啟動(dòng)2. 關(guān)docker hub上有些鏡像的tag被標(biāo)記““This image has vulnerabilities””3. docker-compose中volumes的問題4. javascript - mock.js可以存儲(chǔ)數(shù)據(jù)嗎5. nignx - docker內(nèi)nginx 80端口被占用6. docker安裝后出現(xiàn)Cannot connect to the Docker daemon.7. java - SSH框架中寫分頁(yè)時(shí)service層中不能注入分頁(yè)類8. golang - 用IDE看docker源碼時(shí)的小問題9. docker images顯示的鏡像過多,狗眼被亮瞎了,怎么辦?10. docker api 開發(fā)的端口怎么獲取?

網(wǎng)公網(wǎng)安備