mybatis 乐观锁实现,解决并发问题

情景展示:

银行两操作员同时操作同一账户就是典型的例子。

比如A、B操作员同时读取一余额为1000元的账户,A操作员为该账户增加100元,B操作员同时为该账户扣除50元,A先提交,B后提交。最后实际账户余额为1000-50=950元,但本该为1000+100-50=1050。这就是典型的并发问题。

乐观锁机制在一定程度上解决了这个问题。乐观锁,大多是基于数据版本(Version)记录机制实现。何谓数据版本?即为数据增加一个版本标识,在基于数据库表的版本解决方案中,一般是通过为数据库表增加一个 “version” 字段来实现。

读取出数据时,将此版本号一同读出,之后更新时,对此版本号加一。此时,将提交数据的版本数据与数据库表对应记录的当前版本信息进行比对,如果提交的数据版本号大于数据库表当前版本号,则予以更新,否则认为是过期数据。

对于上面修改用户帐户信息的例子而言,假设数据库中帐户信息表中有一个version字段,当前值为1;而当前帐户余额字段(balance)为1000元。假设操作员A先更新完,操作员B后更新。
a、操作员A此时将其读出(version=1),并从其帐户余额中增加100(1000+100=1100)。
b、在操作员A操作的过程中,操作员B也读入此用户信息(version=1),并从其帐户余额中扣除50(1000-50=950)。
c、操作员A完成了修改工作,将数据版本号加一(version=2),连同帐户增加后余额(balance=1100),提交至数据库更新,此时由于提交数据版本大于数据库记录当前版本,数据被更新,数据库记录version更新为2。
d、操作员B完成了操作,也将版本号加一(version=2)试图向数据库提交数据(balance=950),但此时比对数据库记录版本时发现,操作员B提交的数据版本号为2,数据库记录当前版本也为2,不满足 “提交版本必须大于记录当前版本才能执行更新 “的乐观锁策略,因此,操作员B的提交被驳回。

这样,就避免了操作员B用基于version=1的旧数据修改的结果覆盖操作员A的操作结果的可能。


示例代码:

account建库脚本


 
 
  1. drop table if exists account_wallet;
  2. /*==============================================================*/
  3. /* Table: account_wallet */
  4. /*==============================================================*/
  5. create table account_wallet
  6. (
  7. id int not null comment '用户钱包主键',
  8. user_open_id varchar(64) comment '用户中心的用户唯一编号',
  9. user_amount decimal(10,5),
  10. create_time datetime,
  11. update_time datetime,
  12. pay_password varchar(64),
  13. is_open int comment '0:代表未开启支付密码,1:代表开发支付密码',
  14. check_key varchar(64) comment '平台进行用户余额更改时,首先效验key值,否则无法进行用户余额更改操作',
  15. version int comment '基于mysql乐观锁,解决并发访问'
  16. primary key (id)
  17. );


dao层


 
 
  1. AccountWallet selectByOpenId(String openId);
  2. int updateAccountWallet(AccountWallet record);

service 层


 
 
  1. AccountWallet selectByOpenId(String openId);
  2. int updateAccountWallet(AccountWallet record);

serviceImpl层


 
 
  1. public AccountWallet selectByOpenId(String openId) {
  2. // TODO Auto-generated method stub
  3. return accountWalletMapper.selectByOpenId(openId);
  4. }
  5. public int updateAccountWallet(AccountWallet record) {
  6. // TODO Auto-generated method stub
  7. return accountWalletMapper.updateAccountWallet(record);
  8. }

sql.xml


 
 
  1. <!--通过用户唯一编号,查询用户钱包相关的信息 -->
  2. <select id="selectByOpenId" resultMap="BaseResultMap" parameterType="java.lang.String">
  3. select
  4. <include refid="Base_Column_List" />
  5. from account_wallet
  6. where user_open_id = #{openId,jdbcType=VARCHAR}
  7. </select>
  8. <!--用户钱包数据更改 ,通过乐观锁(version机制)实现 -->
  9. <update id="updateAccountWallet" parameterType="com.settlement.model.AccountWallet">
  10. <![CDATA[
  11. update account_wallet set user_amount = #{userAmount,jdbcType=DECIMAL}, version = version + 1 where id =#{id,jdbcType=INTEGER} and version = #{version,jdbcType=INTEGER}
  12. ]]>
  13. </update>


controller 层


 
 
  1. package com.settlement.controller;
  2. import java.math.BigDecimal;
  3. import javax.servlet.http.HttpServletRequest;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.stereotype.Controller;
  6. import org.springframework.web.bind.annotation.RequestMapping;
  7. import org.springframework.web.bind.annotation.RequestMethod;
  8. import org.springframework.web.bind.annotation.ResponseBody;
  9. import com.settlement.commons.base.BaseController;
  10. import com.settlement.model.AccountWallet;
  11. import com.settlement.service.AccountWalletService;
  12. import com.taobao.api.internal.util.StringUtils;
  13. /**
  14. * 用户钱包Controller
  15. *
  16. * @author zzg
  17. * @date 2017-02-10
  18. */
  19. @Controller
  20. @RequestMapping(value = "/wallet")
  21. public class WalletController extends BaseController {
  22. @Autowired
  23. private AccountWalletService accountWalletService;
  24. /**
  25. * 针对业务系统高并发-----修改用户钱包数据余额,采用乐观锁
  26. *
  27. * @return
  28. */
  29. @RequestMapping(value = "/walleroptimisticlock.action", method = RequestMethod.POST)
  30. @ResponseBody
  31. public String walleroptimisticlock(HttpServletRequest request) {
  32. String result = "";
  33. try {
  34. String openId = request.getParameter("openId") == null ? null
  35. : request.getParameter("openId").trim(); // 用户唯一编号
  36. String openType = request.getParameter("openType") == null ? null
  37. : request.getParameter("openType").trim(); // 1:代表增加,2:代表减少
  38. String amount = request.getParameter("amount") == null ? null
  39. : request.getParameter("amount").trim(); // 金额
  40. if (StringUtils.isEmpty(openId)) {
  41. return "openId is null";
  42. }
  43. if (StringUtils.isEmpty(openType)) {
  44. return "openType is null";
  45. }
  46. if (StringUtils.isEmpty(amount)) {
  47. return "amount is null";
  48. }
  49. AccountWallet wallet = accountWalletService.selectByOpenId(openId);
  50. // 用户操作金额
  51. BigDecimal cash = BigDecimal.valueOf(Double.parseDouble(amount));
  52. cash.doubleValue();
  53. cash.floatValue();
  54. if (Integer.parseInt(openType) == 1) {
  55. wallet.setUserAmount(wallet.getUserAmount().add(cash));
  56. } else if (Integer.parseInt(openType) == 2) {
  57. wallet.setUserAmount(wallet.getUserAmount().subtract(cash));
  58. }
  59. int target = accountWalletService.updateAccountWallet(wallet);
  60. System.out.println("修改用户金额是否:" + (target == 1 ? "成功" : "失败"));
  61. } catch (Exception e) {
  62. result = e.getMessage();
  63. return result;
  64. }
  65. return "success";
  66. }
  67. }


模拟并发访问


 
 
  1. package com.settlement.concurrent;
  2. import java.text.SimpleDateFormat;
  3. import java.util.Date;
  4. import java.util.concurrent.CountDownLatch;
  5. import com.settlement.commons.utils.HttpRequest;
  6. /**
  7. * 模拟用户的并发请求,检测用户乐观锁的性能问题
  8. *
  9. * @author zzg
  10. * @date 2017-02-10
  11. */
  12. public class ConcurrentTest {
  13. final static SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
  14. public static void main(String[] args){
  15. CountDownLatch latch=new CountDownLatch(1);//模拟5人并发请求,用户钱包
  16. for(int i=0;i <5;i++){//模拟5个用户
  17. AnalogUser analogUser = new AnalogUser("user"+i,"58899dcd-46b0-4b16-82df-bdfd0d953bfb","1","20.024",latch);
  18. analogUser.start();
  19. }
  20. latch.countDown();//计数器減一 所有线程释放 并发访问。
  21. System.out.println("所有模拟请求结束 at "+sdf.format(new Date()));
  22. }
  23. static class AnalogUser extends Thread{
  24. String workerName;//模拟用户姓名
  25. String openId;
  26. String openType;
  27. String amount;
  28. CountDownLatch latch;
  29. public AnalogUser(String workerName, String openId, String openType, String amount,
  30. CountDownLatch latch) {
  31. super();
  32. this.workerName = workerName;
  33. this.openId = openId;
  34. this.openType = openType;
  35. this.amount = amount;
  36. this.latch = latch;
  37. }
  38. @Override
  39. public void run() {
  40. // TODO Auto-generated method stub
  41. try {
  42. latch.await(); //一直阻塞当前线程,直到计时器的值为0
  43. } catch (InterruptedException e) {
  44. e.printStackTrace();
  45. }
  46. post();//发送post 请求
  47. }
  48. public void post(){
  49. String result = "";
  50. System.out.println("模拟用户: "+workerName+" 开始发送模拟请求 at "+sdf.format(new Date()));
  51. result = HttpRequest.sendPost("http://localhost:8080/Settlement/wallet/walleroptimisticlock.action", "openId="+openId+"&openType="+openType+"&amount="+amount);
  52. System.out.println("操作结果:"+ result);
  53. System.out.println("模拟用户: "+workerName+" 模拟请求结束 at "+sdf.format(new Date()));
  54. }
  55. }
  56. }



转载自:https://blog.csdn.net/zhouzhiwengang/article/details/54973509

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值