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

java樂觀鎖原理與實現(xiàn)案例分析

 更新時間:2019年10月31日 08:41:50   作者:zhangdehua678  
這篇文章主要介紹了java樂觀鎖原理與實現(xiàn),結合具體案例形式分析了樂觀鎖的原理及java使用樂觀鎖實現(xiàn)自動派單功能的相關操作技巧,需要的朋友可以參考下

本文實例講述了java樂觀鎖原理與實現(xiàn)。分享給大家供大家參考,具體如下:

簡單說說樂觀鎖。樂觀鎖是相對于悲觀鎖而言。悲觀鎖認為,這個線程,發(fā)生并發(fā)的可能性極大,線程沖突幾率大,比較悲觀。一般用synchronized實現(xiàn),保證每次操作數(shù)據(jù)不會沖突。樂觀鎖認為,線程沖突可能性小,比較樂觀,直接去操作數(shù)據(jù),如果發(fā)現(xiàn)數(shù)據(jù)已經(jīng)被更改(通過版本號控制),則不更新數(shù)據(jù),再次去重復 所需操作,知道沒有沖突(使用遞歸算法)。

因為樂觀鎖使用遞歸+版本號控制  實現(xiàn),所以,如果線程沖突幾率大,使用樂觀鎖會重復很多次操作(包括查詢數(shù)據(jù)庫),尤其是遞歸部分邏輯復雜,耗時和耗性能,是低效不合適的,應考慮使用悲觀鎖。

樂觀鎖悲觀鎖的選擇:

  • 樂觀鎖:并發(fā)沖突幾率小,對應模塊遞歸操作簡單    時使用
  • 悲觀鎖:并發(fā)幾率大,對應模塊操作復雜 時使用

下面給出一個樂觀鎖實例:

/**
 * 自動派單
 * 只查出一條  返回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; // 默認先進先審
        }
    }
    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;
    }
}

對應更新的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>

簡要說明:表設計時,需要往表里加一個version字段。每次查詢時,查出帶有version的數(shù)據(jù)記錄,更新數(shù)據(jù)時,判斷數(shù)據(jù)庫里對應id的記錄的version是否和查出的version相同。若相同,則更新數(shù)據(jù)并把版本號+1;若不同,則說明,該數(shù)據(jù)發(fā)送并發(fā),被別的線程使用了,進行遞歸操作,再次執(zhí)行遞歸方法,知道成功更新數(shù)據(jù)為止。

上述automaticAssign方法即實現(xiàn)了一個樂觀鎖,作用是沖數(shù)據(jù)庫里更新一條數(shù)據(jù)病返回前端。如果并發(fā)率大,一次請求可能則會重復執(zhí)行很多次automaticAssign,則性能低。如果并發(fā)很樂觀,用戶請求少,則不需要用synchronized,多線程時性能高。

在此只是簡單說說,詳細概念等需另行查閱相關資料。

更多java相關內容感興趣的讀者可查看本站專題:《Java面向對象程序設計入門與進階教程》、《Java數(shù)據(jù)結構與算法教程》、《Java操作DOM節(jié)點技巧總結》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總

希望本文所述對大家java程序設計有所幫助。

相關文章

最新評論