java樂觀鎖原理與實(shí)現(xiàn)案例分析
本文實(shí)例講述了java樂觀鎖原理與實(shí)現(xiàn)。分享給大家供大家參考,具體如下:
簡單說說樂觀鎖。樂觀鎖是相對于悲觀鎖而言。悲觀鎖認(rèn)為,這個線程,發(fā)生并發(fā)的可能性極大,線程沖突幾率大,比較悲觀。一般用synchronized實(shí)現(xiàn),保證每次操作數(shù)據(jù)不會沖突。樂觀鎖認(rèn)為,線程沖突可能性小,比較樂觀,直接去操作數(shù)據(jù),如果發(fā)現(xiàn)數(shù)據(jù)已經(jīng)被更改(通過版本號控制),則不更新數(shù)據(jù),再次去重復(fù) 所需操作,知道沒有沖突(使用遞歸算法)。
因?yàn)闃酚^鎖使用遞歸+版本號控制 實(shí)現(xiàn),所以,如果線程沖突幾率大,使用樂觀鎖會重復(fù)很多次操作(包括查詢數(shù)據(jù)庫),尤其是遞歸部分邏輯復(fù)雜,耗時和耗性能,是低效不合適的,應(yīng)考慮使用悲觀鎖。
樂觀鎖悲觀鎖的選擇:
- 樂觀鎖:并發(fā)沖突幾率小,對應(yīng)模塊遞歸操作簡單 時使用
- 悲觀鎖:并發(fā)幾率大,對應(yīng)模塊操作復(fù)雜 時使用
下面給出一個樂觀鎖實(shí)例:
/** * 自動派單 * 只查出一條 返回list只是為了和查詢接口統(tǒng)一 * 視頻審核訂單不派送 * @param paramMap * @return */ public List<AutomaticAssignDto> automaticAssign(Map<String, Object> paramMap){ //派送規(guī)則 String changeSortSet = RedisCacheUtil.getValue(CACHE_TYPE.APP, "changeSortSet"); if (StringUtils.isBlank(changeSortSet)) { changeSortSet = customerManager.getDictionaryByCode("changeSortSet"); if (StringUtils.isNotBlank(changeSortSet)) { RedisCacheUtil.addValue(CACHE_TYPE.APP, "changeSortSet", changeSortSet,30,TimeUnit.DAYS); } else { changeSortSet = ConstantsUtil.AssignRule.FIFO; // 默認(rèn)先進(jìn)先審 } } AutomaticAssignDto automaticAssignDto = new AutomaticAssignDto(); automaticAssignDto.setChangeSortSet(changeSortSet); automaticAssignDto.setUserTeam(CommonUtils.getValue(paramMap, "userTeam")); List<AutomaticAssignDto> waitCheckList = automaticAssignMybatisDao.automaticAssignOrder(automaticAssignDto); if(waitCheckList != null && waitCheckList.size()>0){ automaticAssignDto = waitCheckList.get(0); automaticAssignDto.setSendStatus(ConstantsUtil.SendStatus.SEND); automaticAssignDto.setBindTime(new Date()); automaticAssignDto.setUserId(Long.parseLong(paramMap.get("userId").toString()) ); int sum = automaticAssignMybatisDao.bindAutomaticAssignInfo(automaticAssignDto); if(sum == 1){ return waitCheckList; }else{ //已被更新 則再次獲取 return automaticAssign(paramMap); } }else{ return null; } }
對應(yīng)更新的sql:
<update id="bindAutomaticAssignInfo" parameterType="com.star.manager.dto.apply.AutomaticAssignDto"> UPDATE t_automatic_assign SET SEND_STATUS = #{sendStatus} , BIND_TIME = SYSDATE() , LOCKED_FINISHTIME = SYSDATE(), USER_ID = #{userId} , VERSION = VERSION + 1, UPDATE_DATE = SYSDATE() WHERE SLT_ACCOUNT_ID = #{sltAccountId} AND VERSION = #{version} </update>
簡要說明:表設(shè)計時,需要往表里加一個version字段。每次查詢時,查出帶有version的數(shù)據(jù)記錄,更新數(shù)據(jù)時,判斷數(shù)據(jù)庫里對應(yīng)id的記錄的version是否和查出的version相同。若相同,則更新數(shù)據(jù)并把版本號+1;若不同,則說明,該數(shù)據(jù)發(fā)送并發(fā),被別的線程使用了,進(jìn)行遞歸操作,再次執(zhí)行遞歸方法,知道成功更新數(shù)據(jù)為止。
上述automaticAssign方法即實(shí)現(xiàn)了一個樂觀鎖,作用是沖數(shù)據(jù)庫里更新一條數(shù)據(jù)病返回前端。如果并發(fā)率大,一次請求可能則會重復(fù)執(zhí)行很多次automaticAssign,則性能低。如果并發(fā)很樂觀,用戶請求少,則不需要用synchronized,多線程時性能高。
在此只是簡單說說,詳細(xì)概念等需另行查閱相關(guān)資料。
更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Java面向?qū)ο蟪绦蛟O(shè)計入門與進(jìn)階教程》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點(diǎn)技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對大家java程序設(shè)計有所幫助。
相關(guān)文章
SpringBoot項(xiàng)目Jar包如何瘦身部署的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot項(xiàng)目Jar包如何瘦身部署的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09Java的JDBC中Statement與CallableStatement對象實(shí)例
這篇文章主要介紹了Java的JDBC中Statement與CallableStatement對象實(shí)例,JDBC是Java編程中用于操作數(shù)據(jù)庫的API,需要的朋友可以參考下2015-12-12SpringMvc MultipartFile實(shí)現(xiàn)圖片文件上傳示例
本篇文章主要介紹了SpringMvc MultipartFile實(shí)現(xiàn)圖片文件上傳示例,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-02-02springmvc中RequestMappingHandlerAdapter與HttpMessageConverter的
今天小編就為大家分享一篇關(guān)于springmvc中RequestMappingHandlerAdapter與HttpMessageConverter的裝配講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-01-01Java中JVM的雙親委派、內(nèi)存溢出、垃圾回收和調(diào)優(yōu)詳解
這篇文章主要介紹了Java中JVM的雙親委派、內(nèi)存溢出、垃圾回收和調(diào)優(yōu)詳解,類加載器是Java虛擬機(jī)(JVM)的一個重要組成部分,它的主要作用是將類的字節(jié)碼加載到內(nèi)存中,并生成對應(yīng)的Class對象,需要的朋友可以參考下2023-07-07一文搞懂java中類及static關(guān)鍵字執(zhí)行順序
這篇文章主要介紹了一文搞懂java中類及static關(guān)鍵字執(zhí)行順序,文章通過類的生命周期展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-09-09springboot使用DynamicDataSource動態(tài)切換數(shù)據(jù)源的實(shí)現(xiàn)過程
這篇文章主要給大家介紹了關(guān)于springboot使用DynamicDataSource動態(tài)切換數(shù)據(jù)源的實(shí)現(xiàn)過程,Spring Boot應(yīng)用中可以配置多個數(shù)據(jù)源,并根據(jù)注解靈活指定當(dāng)前使用的數(shù)據(jù)源,需要的朋友可以參考下2023-08-08