我的编程空间,编程开发者的网络收藏夹
学习永远不晚

mybatis使用乐观锁和悲观锁

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

mybatis使用乐观锁和悲观锁

悲观锁和乐观锁的概念:

悲观锁:就是独占锁,不管读写都上锁了。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。Java中synchronized和ReentrantLock等独占锁就是悲观锁思想的实现。

乐观锁:不上锁,读取的时候带版本号,写入的时候带着这个版本号,如果不一致就失败,乐观锁适用于多读的应用类型,因为写多的时候会经常失败。

1 Maven依赖
需要引入spring-boot-starter-data-jpa,这里要访问数据库,所以要依赖数据库相关jar包。

<dependency>            <groupId>org.springframework.boot</groupId>            <artifactId>spring-boot-starter-web</artifactId>        </dependency>        <dependency>            <groupId>org.mybatis.spring.boot</groupId>            <artifactId>mybatis-spring-boot-starter</artifactId>        </dependency>        <dependency>            <groupId>mysql</groupId>            <artifactId>mysql-connector-java</artifactId>        </dependency>        <dependency>            <groupId>org.apache.commons</groupId>            <artifactId>commons-dbcp2</artifactId>        </dependency>

2 配置文件
在application.properties 中需要添加下面的配置:

spring.datasource.type=org.apache.commons.dbcp2.BasicDataSourcespring.datasource.dbcp2.max-wait-millis=60000spring.datasource.dbcp2.min-idle=20spring.datasource.dbcp2.initial-size=2spring.datasource.dbcp2.validation-query=SELECT 1spring.datasource.dbcp2.connection-properties=characterEncoding=utf8spring.datasource.dbcp2.validation-query=SELECT 1spring.datasource.dbcp2.test-while-idle=truespring.datasource.dbcp2.test-on-borrow=truespring.datasource.dbcp2.test-on-return=falsespring.datasource.driverClassName = com.mysql.jdbc.Driverspring.datasource.url=jdbc:mysql://127.0.0.1:3306/boot?useUnicode=true&characterEncoding=utf8&serverTimezone=UTCspring.datasource.username=cffspring.datasource.password=123456mybatis.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl

这里面,包含了数据库连接信息、数据源的连接池配置信息、mybatis配置信息。

spring.datasource.dbcp2是配置dbcp2的连接池信息;
spring.datasource.type指明数据源的类型;
最上面的spring.datasource.xxx指明数据库连接池信息;
mybatis.configuration.log-impl指明mybatis的日志打印方式
三、悲观锁
悲观锁在数据库的访问中使用,表现为:前一次请求没执行完,后面一个请求就一直在等待。

1 Dao层
数据库要实现悲观锁,就是将sql语句带上for update即可。 for update 是行锁

所在mybatis的查询sql加上for update,就实现了对当前记录的锁定,就实现了悲观锁。

UserInfoDao :

package com.cff.springbootwork.mybatislock.dao;import org.apache.ibatis.annotations.Insert;import org.apache.ibatis.annotations.Mapper;import org.apache.ibatis.annotations.Param;import org.apache.ibatis.annotations.Select;import org.apache.ibatis.annotations.Update;import cn.pomit.springwork.mybatislock.domain.UserInfo;@Mapperpublic interface UserInfoDao {    @Select({        ""})    UserInfo findByUserNameForUpdate(@Param("userName") String userName);    @Update({        ""    })    int update(UserInfo userInfo);    @Insert({        ""    })    int save(UserInfo entity);}

这里,findByUserNameForUpdate的sql中加上了for update。update就是普通的更新而已。

2 Service层
更新数据库前,先调用findByUserNameForUpdate方法,使上面的配置的悲观锁锁定表记录,然后再更新。

UserInfoService :

package com.cff.springbootwork.mybatislock.service;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Service;import org.springframework.transaction.annotation.Transactional;import org.springframework.util.StringUtils;import cn.pomit.springwork.mybatislock.domain.UserInfo;import cn.pomit.springwork.mybatislock.mapper.UserInfoDao;@Servicepublic class UserInfoService {    @Autowired    UserInfoDao userInfoDao;    public void save(UserInfo entity) {        entity.setVersion(0);        userInfoDao.save(entity);    }    @Transactional    public UserInfo getUserInfoByUserNamePessimistic(String userName) {        return userInfoDao.findByUserNameForUpdate(userName);    }    @Transactional    public void updateWithTimePessimistic(UserInfo entity, int time) throws InterruptedException {              UserInfo userInfo = userInfoDao.findByUserNameForUpdate(entity.getUserName());        if (userInfo == null)            return;        if (!StringUtils.isEmpty(entity.getMobile())) {            userInfo.setMobile(entity.getMobile());        }        if (!StringUtils.isEmpty(entity.getName())) {            userInfo.setName(entity.getName());        }        Thread.sleep(time * 1000L);        userInfoDao.update(userInfo);    }    @Transactional    public void updatePessimistic(UserInfo entity) {        UserInfo userInfo = userInfoDao.findByUserNameForUpdate(entity.getUserName());        if (userInfo == null)            return;        if (!StringUtils.isEmpty(entity.getMobile())) {            userInfo.setMobile(entity.getMobile());        }        if (!StringUtils.isEmpty(entity.getName())) {            userInfo.setName(entity.getName());        }        userInfoDao.update(userInfo);    }}

测试中,我们在update方法中sleep几秒,其他线程的update将一直等待。

3 测试Web层
可以先调用/update/{time}接口,延迟执行,然后马上调用/update接口,会发现,/update接口一直在等待/update/{time}接口执行完成。

MybatisPessLockRest :

package com.cff.springbootwork.mybatislock.web;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.web.bind.annotation.PathVariable;import org.springframework.web.bind.annotation.RequestBody;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestMethod;import org.springframework.web.bind.annotation.RestController;import cn.pomit.springwork.mybatislock.domain.UserInfo;import cn.pomit.springwork.mybatislock.service.UserInfoService;@RestController@RequestMapping("/mybatispesslock")public class MybatisPessLockRest {    @Autowired    UserInfoService userInfoService;    @RequestMapping(value = "/detail/{name}", method = { RequestMethod.GET })    public UserInfo detail(@PathVariable("name") String name) {        return userInfoService.getUserInfoByUserNamePessimistic(name);    }    @RequestMapping(value = "/save")    public String save(@RequestBody UserInfo userInfo) throws InterruptedException {        userInfoService.save(userInfo);        return "0000";    }    @RequestMapping(value = "/update/{time}")    public String update(@RequestBody UserInfo userInfo, @PathVariable("time") int time) throws InterruptedException {        userInfoService.updateWithTimePessimistic(userInfo, time);        return "0000";    }    @RequestMapping(value = "/update")    public String update(@RequestBody UserInfo userInfo) throws InterruptedException {        userInfoService.updatePessimistic(userInfo);        return "0000";    }}

四、乐观锁
数据库访问dao层还是3.1那个UserInfoDao。

1 Dao层
UserInfoDao更新时,需要携带version字段进行更新:and version = #{version}。如果version不一致,是不会更新成功的,这时候,我们的select查询是不能带锁的。

UserInfoDao :

package com.cff.springbootwork.mybatislock.dao;import org.apache.ibatis.annotations.Insert;import org.apache.ibatis.annotations.Mapper;import org.apache.ibatis.annotations.Param;import org.apache.ibatis.annotations.Select;import org.apache.ibatis.annotations.Update;import cn.pomit.springwork.mybatislock.domain.UserInfo;@Mapperpublic interface UserInfoDao {    @Select({        ""})    UserInfo findByUserName(@Param("userName") String userName);    @Update({        ""    })    int updateWithVersion(UserInfo userInfo);    @Insert({        ""    })    int save(UserInfo entity);}

2 Service层
service层我们做一下简单的调整。更新数据库前,先调用findByUserName方法,查询出当前的版本号,然后再更新。

UserInfoService :

package com.cff.springbootwork.mybatislock.service;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Service;import org.springframework.transaction.annotation.Transactional;import org.springframework.util.StringUtils;import cn.pomit.springwork.mybatislock.domain.UserInfo;import cn.pomit.springwork.mybatislock.mapper.UserInfoDao;@Servicepublic class UserInfoService {    @Autowired    UserInfoDao userInfoDao;    public UserInfo getUserInfoByUserName(String userName){        return userInfoDao.findByUserName(userName);    }    public void save(UserInfo entity) {        entity.setVersion(0);        userInfoDao.save(entity);    }    @Transactional    public void updateWithTimeOptimistic(UserInfo entity, int time) throws Exception {              UserInfo userInfo = userInfoDao.findByUserName(entity.getUserName());        if (userInfo == null)            return;        if (!StringUtils.isEmpty(entity.getMobile())) {            userInfo.setMobile(entity.getMobile());        }        if (!StringUtils.isEmpty(entity.getName())) {            userInfo.setName(entity.getName());        }        Thread.sleep(time * 1000L);        int ret = userInfoDao.updateWithVersion(userInfo);        if(ret < 1)throw new Exception("乐观锁导致保存失败");    }    @Transactional    public void updateOptimistic(UserInfo entity) throws Exception {        UserInfo userInfo = userInfoDao.findByUserName(entity.getUserName());        if (userInfo == null)            return;        if (!StringUtils.isEmpty(entity.getMobile())) {            userInfo.setMobile(entity.getMobile());        }        if (!StringUtils.isEmpty(entity.getName())) {            userInfo.setName(entity.getName());        }        int ret = userInfoDao.updateWithVersion(userInfo);        if(ret < 1)throw new Exception("乐观锁导致保存失败");    }}

2 测试Web层
可以先调用/update/{time}接口,延迟执行,然后马上调用/update接口,会发现,/update接口不会等待/update/{time}接口执行完成,读取完版本号能够成功更新数据,但是/update/{time}接口等待足够时间以后,更新的时候会失败,因为它的版本和数据库的已经不一致了。

注意: 这里更新失败不会抛异常,但是返回值会是0,即更新不成功,需要自行判断。jpa的乐观锁可以抛出异常,手动catch到再自行处理。

MybatisOptiLockRest :

package com.cff.springbootwork.mybatislock.web;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.web.bind.annotation.PathVariable;import org.springframework.web.bind.annotation.RequestBody;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RequestMethod;import org.springframework.web.bind.annotation.RestController;import cn.pomit.springwork.mybatislock.domain.UserInfo;import cn.pomit.springwork.mybatislock.service.UserInfoService;@RestController@RequestMapping("/mybatislock")public class MybatisOptiLockRest {    @Autowired    UserInfoService userInfoService;    @RequestMapping(value = "/detail/{name}", method = { RequestMethod.GET })    public UserInfo detail(@PathVariable("name") String name) {        return userInfoService.getUserInfoByUserName(name);    }    @RequestMapping(value = "/save")    public String save(@RequestBody UserInfo userInfo) throws InterruptedException {        userInfoService.save(userInfo);        return "0000";    }    @RequestMapping(value = "/update/{time}")    public String update(@RequestBody UserInfo userInfo, @PathVariable("time") int time) throws Exception {        userInfoService.updateWithTimeOptimistic(userInfo, time);        return "0000";    }    @RequestMapping(value = "/update")    public String update(@RequestBody UserInfo userInfo) throws Exception {        userInfoService.updateOptimistic(userInfo);        return "0000";    }}

来源地址:https://blog.csdn.net/qq_43459184/article/details/132582876

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

mybatis使用乐观锁和悲观锁

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

mybatis使用乐观锁和悲观锁

悲观锁和乐观锁的概念: 悲观锁:就是独占锁,不管读写都上锁了。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。Java中synchronized和ReentrantLock等独占锁就是悲
2023-08-30

mysql 乐观锁和悲观锁的具体使用

本文主要介绍了mysql 乐观锁和悲观锁的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2023-01-08

mysql 乐观锁和悲观锁的具体使用

目录悲观锁介绍(百科):1如果不采用锁,那么操作方法如下:2使用悲观锁来实现:补充:mysql select…for update的Row Lock与Table Lock乐观锁介绍:使用举例:以MySQL InnoDB为例悲观
2023-01-09

redis乐观锁与悲观锁怎么使用

本篇内容主要讲解“redis乐观锁与悲观锁怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“redis乐观锁与悲观锁怎么使用”吧!概念Redis是一个内存中的键值存储系统,支持多种数据结构,
2023-07-06

mysql悲观锁和乐观锁如何实现

悲观锁是在对数据进行操作之前就先加锁,防止其他事务对数据进行修改,从而确保数据的一致性。在MySQL中,可以通过使用SELECT … FOR UPDATE语句来实现悲观锁。例如:START TRANSACTION;SELECT * FR
mysql悲观锁和乐观锁如何实现
2024-05-08

MySQL怎么处理乐观锁和悲观锁

MySQL中可以通过以下两种方式来处理乐观锁和悲观锁:乐观锁:在MySQL中,可以通过使用版本号或时间戳来实现乐观锁。在数据库表中增加一个版本号或者时间戳字段,每次对数据进行修改时,先查询该字段的值,然后在更新数据时将该字段的值加1或者更新
MySQL怎么处理乐观锁和悲观锁
2024-04-09

合理的使用MySQL乐观锁与悲观锁

针对 MySQL的乐观锁与悲观锁的使用,基本都是按照业务场景针对性使用的。针对每个业务场景,对应的使用锁。 但是两种锁无非都是解决并发所产生的问题。下面我们来看看如何合理的使用乐观锁与悲观锁   何为悲观锁 悲观锁(Pessimistic Lock):就是很悲
合理的使用MySQL乐观锁与悲观锁
2015-02-12

MySQL中的悲观锁与乐观锁

在关系型数据库中,悲观锁与乐观锁是解决资源并发场景的解决方案,接下来将详细讲解一下这两个并发解决方案的实际使用及优缺点。 首先定义一下数据库,做一个最简单的库存表,如下设计:CREATE TABLE `order_stock` (`id`
2022-05-15

编程热搜

目录