溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

怎么使用MyBatisPlus+SpringBoot實現(xiàn)樂觀鎖功能

發(fā)布時間:2023-03-24 13:43:32 來源:億速云 閱讀:123 作者:iii 欄目:開發(fā)技術(shù)

今天小編給大家分享一下怎么使用MyBatisPlus+SpringBoot實現(xiàn)樂觀鎖功能的相關(guān)知識點,內(nèi)容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

    一、商城數(shù)據(jù)不一致的場景

    如果商城中有一件商品,成本價是80元,售價是100元。經(jīng)理先是通知小李,說你去把商品價格增加50元。小李正在玩游戲,耽擱了一個小時。正好一個小時后,經(jīng)理覺得商品價格增加到150元,價格太高,可能會影響銷量。又通知小王,你把商品價格降低30元。

    此時,小李和小王同時操作商品后臺系統(tǒng)。小李操作的時候,系統(tǒng)先取出商品價格100元;小王也在操作,取出的商品價格也是100元。小李將價格加了50元,并將100+50=150元存入了數(shù)據(jù)庫;小王將商品減了30元,并將100-30=70元存入了數(shù)據(jù)庫。是的,如果沒有鎖,小李的操作就完全被小王的覆蓋了。

    現(xiàn)在商品價格是70元,比成本價低10元。幾分鐘后,這個商品很快出售了1千多件商品,老板虧1萬多。

    二、演示這一過程

    1、數(shù)據(jù)庫中增加商品表

    CREATE TABLE product
    (
        id BIGINT(20) NOT NULL AUTO_INCREMENT COMMENT '主鍵ID',
        name VARCHAR(30) NULL DEFAULT NULL COMMENT '商品名稱',
        price INT(11) DEFAULT 0 COMMENT '價格',
        version INT(11) DEFAULT 0 COMMENT '樂觀鎖版本號',
        PRIMARY KEY (id)
    );
    
    INSERT INTO product (id, NAME, price) VALUES (1, '筆記本', 100);

    2、創(chuàng)建實體類

    @Data
    public class Product {
        private Long id;
        private String name;
        private Integer price;
        private Integer version;
    }

    3、創(chuàng)建Mapper

    public interface ProductMapper extends BaseMapper<Product> {
        
    }

    4、進行測試

    @RunWith(SpringRunner.class)
    @SpringBootTest
    public class ProductVersionTest {
    	@Resource
    	private ProductMapper productMapper;
    	
    	@Test
    	public void testProductUpdate() {
    	
    	    //1、小李
    	    Product p1 = productMapper.selectById(1L);
    	
    	    //2、小王
    	    Product p2 = productMapper.selectById(1L);
    	
    	    //3、小李將價格加了50元,存入了數(shù)據(jù)庫
    	    p1.setPrice(p1.getPrice() + 50);
    	    int result1 = productMapper.updateById(p1);
    	    System.out.println("小李修改結(jié)果:" + result1);
    	
    	    //4、小王將商品減了30元,存入了數(shù)據(jù)庫
    	    p2.setPrice(p2.getPrice() - 30);
    	    int result2 = productMapper.updateById(p2);
    	    System.out.println("小王修改結(jié)果:" + result2);
    	
    	    //最后的結(jié)果
    	    Product p3 = productMapper.selectById(1L);
    	    System.out.println("最后的結(jié)果:" + p3.getPrice());
    	}
    }

    最后輸出的是 70元,與經(jīng)理預(yù)期的120元不同,導(dǎo)致虧損,如何防止這樣的異常發(fā)生,解決方案是使用樂觀鎖

    三、樂觀鎖方案

    數(shù)據(jù)庫中添加version字段:取出記錄時,獲取當前version

    SELECT id,`name`,price,`version` FROM product WHERE id=1

    更新時,version + 1,如果where語句中的version版本不對,則更新失敗

    UPDATE product SET price=price+50, `version`=`version` + 1 WHERE id=1 AND `version`=1

    四、樂觀鎖實現(xiàn)流程

    1、修改實體類

    添加 @Version 注解

    @Version
    private Integer version;

    2、添加樂觀鎖插件

    @Configuration
    //@MapperScan("com.koo.modules.*.dao")
    public class MybatisPlusConfig {
    
        /**
         * 新的分頁插件,一緩和二緩遵循mybatis的規(guī)則,需要設(shè)置 MybatisConfiguration#useDeprecatedExecutor = false 避免緩存出現(xiàn)問題(該屬性會在舊插件移除后一同移除)
         */
        @Bean
        public MybatisPlusInterceptor mybatisPlusInterceptor() {
            MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
            interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
            //實現(xiàn)樂觀鎖,保證數(shù)據(jù)的準確性
            interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor());
            return interceptor;
        }
    
        @Bean
        public ConfigurationCustomizer configurationCustomizer() {
            return configuration -> configuration.setUseDeprecatedExecutor(false);
        }
    
    }

    3、優(yōu)化流程

    (判斷第二次更新數(shù)據(jù)是否成功,不成功則重新取數(shù)據(jù)進行更新)

    @RunWith(SpringRunner.class)
    @SpringBootTest
    public class ProductVersionTest {
    	@Resource
    	private ProductMapper productMapper;
    	
    	@Test
    	public void testProductUpdate() {
    	
    	    //1、小李
    	    Product p1 = productMapper.selectById(1L);
    	
    	    //2、小王
    	    Product p2 = productMapper.selectById(1L);
    	
    	    //3、小李將價格加了50元,存入了數(shù)據(jù)庫
    	    p1.setPrice(p1.getPrice() + 50);
    	    int result1 = productMapper.updateById(p1);
    	    System.out.println("小李修改結(jié)果:" + result1);
    	
    	    //4、小王將商品減了30元,存入了數(shù)據(jù)庫
    	    p2.setPrice(p2.getPrice() - 30);
    	    int result2 = productMapper.updateById(p2);
    	    System.out.println("小王修改結(jié)果:" + result2);
    	
    		if(result2 == 0){//更新失敗,重試
    			    System.out.println("小王重試");
    			    //重新獲取數(shù)據(jù)
    			    p2 = productMapper.selectById(1L);
    			    //更新
    			    p2.setPrice(p2.getPrice() - 30);
    			    productMapper.updateById(p2);
    		}
    	    //最后的結(jié)果
    	    Product p3 = productMapper.selectById(1L);
    	    System.out.println("最后的結(jié)果:" + p3.getPrice());
    	}
    }

    輸出結(jié)果為120,數(shù)據(jù)正確

    以上就是“怎么使用MyBatisPlus+SpringBoot實現(xiàn)樂觀鎖功能”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學(xué)習更多的知識,請關(guān)注億速云行業(yè)資訊頻道。

    向AI問一下細節(jié)

    免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

    AI