java中事务中遇到锁会造成什么问题,以及该如何解决?

在spring中实现事务有多种方式,主要是两种:一种是声明式事务,一种是编程式事务,今天我们就讲声明式事务中的一种,使用注解@Transactional,这个注解的作用就是帮助我们在代码执行完毕之后自动提交事务,在发生异常时进行回滚。

比如以下这段代码:

@Transactional
public void updateDBInfo(List<Entity> entityList){//待更新的数据entityList.stream().foreach((arg) -> {arg.setModifier("xxx");arg.setModifierTime("xxxx-xx-xx hh:mm:ss");});this.updateBatchById(entityList);
}

 这是一段非常简单的代码段。假如我们就是需要加进行更新,但是如果这段代码需要涉及到锁的话,如果架构是分布式,也就是部署了多台service Application服务器,然后进行负债均衡,那么此时如果有多个用户同时操作的话,每个用户的请求会由于负债均衡可能到达不同的service Application服务器应用。所以这时候我们的锁用线程级别的锁是不行的,只能用进程级别的锁,也就是分布式锁。分布式锁的话可以用redis的那个命令叫setNx,这里我就不过多赘述了。主要讲在事务里面有锁会造成什么问题,以及如何解决该问题。

如果我们还是使用这个注解@Transactional,这个注解我们刚刚也说了。这个注解的作用就是帮助我们在代码执行完毕之后自动提交事务,在发生异常时进行回滚。

首先改造代码进行加锁:

@Transactional
public void updateDBInfo(List<Entity> entityList){
RLock lock = null;
try {lock = redissonTool.tryLock(RedisKeyConstants.REDIS_KEY_PURCHASE_ORDER_EXTEND, Codes.KEY_ERROR);//待更新的数据entityList.stream().foreach((arg) -> {arg.setModifier("xxx");arg.setModifierTime("xxxx-xx-xx hh:mm:ss");});this.updateBatchById(entityList);
} finally {redissonTool.unLock(lock);
}
}

 如果我们是这样写的话,那么会有什么问题呢?

当代码执行完毕,也就是finally里面的锁都释放了,去提交事务,但是提交事务也是需要时间的,有可能还没提交事务完,由于锁释放了,其他客户端就会拿到锁了,但是此时该用户感知不到此时DB数据库中的数据是最新的,也就是所谓的可见性,什么是可见性?

可见性:顾名思义就是一个线程的操作结果对其他所有线程都是可见的,因为每一个线程都共享同一个共享主内存。

那么这时候另一个用户在客户端就不知道这个DB的数据 是更改的,又会去更改,然后提交事务,这时候刚刚那个客户端的用户也提交事务成功了,那么就会造成脏数据了,数据不一致。

我给大家画个图展示:

但是由于我刚刚说的这种,A还没提交更新事务之后,锁释放,B拿到,B更新完,提交更新事务,成功改为2,A的提交更新事务完成。此时就还是1.就是脏数据了。

所以有什么办法解决呢?

那我们把自动事务注解@Transactional改成手动事务。让事务提交成功之后再去释放锁,而不是在锁释放之后再提交事务。

手动事务我用的是transactionTemplate.execute(TransactionCallback callback);然后用的是TransactionCallback接口的抽象实现类,用匿名内部类。

TransactionCallbackWithoutResult的抽象方法:doInTransactionWithoutResult

也就是:

transactionTemplate.execute(new TransactionCallbackWithoutResult() {@Overrideprotected void doInTransactionWithoutResult(TransactionStatus status) {//业务更新逻辑}});

总结:

        在事务中碰到锁,必须使用手动事务,让事务提交成功之后,再去释放锁。而非释放锁之后再提交事务。也就是不能使用@Transactional注解的自动提交事务。

一般模板:

RLock rLock = null;
try{rLock =  redissonTool.tryLock("REDIS_LOCK_NAME");
}catch(){transactionTemplate.execute(new TransactionCallbackWithoutResult(){protected void doInTransctionWithoutResult(){//业务更新逻辑;}})
}finally{rlock.unlock();
}

 最后:

        如果大家觉得这篇文章对你们有所帮助的话,麻烦点个免费的赞赞,谢谢,也祝各位码农在未来的IT道路上越走越远。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/23247.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

淘宝评论数据爬取全攻略

一、淘宝评论数据爬取的背景与意义 随着互联网的快速发展&#xff0c;电子商务平台如淘宝、京东等在我国市场占有率逐年上升。消费者在购买商品时&#xff0c;除了关注商品的价格、质量等因素外&#xff0c;还会参考其他消费者的评价和评论。淘宝评论数据爬取是指通过技术手段…

C# NX二次开发-设置背景颜色

使用UF函数能直接设置UG背景颜色: 1.设置背景颜色选项为纯色: 2.编写更新背景颜色代码: var nxColor NXColor.Factory._Get(186);var rgb nxColor.GetRgb();double[] arr [rgb.R, rgb.G, rgb.B];theUf.Disp.SetColor(UFConstants.UF_DISP_BACKGROUND_COLOR, UFConstants.UF…

flask轻松入门,概念讲解

Hello World Flask 是轻量级web框架&#xff0c;仅保留了核心功能&#xff1a; 请求响应处理模板渲染URL路由 文章目录 Hello Worldflask命令模式python命令模式两种模式对比修改入口文件配置flask命令修改python命令修改 修改端口和地址flask命令修改python命令修改 修改 URL …

java——顺序表

前言&#xff1a;顺序表是线性表的一种&#xff0c;它是较于数组更加灵活的一种储存方式。线性表通常是逻辑上是连续的一条直线&#xff0c;但在物理上不是连续的。java中已经实现好了一个顺序表&#xff0c;搭配泛型可以支持各种类型的使用&#xff0c;下面就来介绍该如何使用…

Mysql疑难报错排查 - Field ‘XXX‘ doesn‘t have a default value

项目场景&#xff1a; 数据库环境 &#xff1a;mysql8; 工程使用&#xff1a;MyBatisPlus 表情况&#xff1a; 问题描述 某一个插入语句使用了 MyBatisPlus 的 save 方法&#xff0c;因为end_time1 end_time2都并没有值&#xff0c;所以在MyBatisPlus默认情况下&#xff0c;…

如何使自己写的代码易读易懂?

〓● 如果代码可读性不佳、不容易理解&#xff0c;可能造成如下问题&#xff1a; 〓❏ 其他工程师浪费时间解读它&#xff1b; 〓❏ 误解导致引入缺陷&#xff1b; 〓❏ 其他工程师修改时破坏代码。 〓● 提高代码可读性&#xff0c;有时候可能使其变得更为冗长、占用更多的…

【Python】深入认识Python数据类型和变量

​​​​ 文章目录 1. 引言数据类型的重要性Python中的数据类型概述 2. 数字类型整型&#xff08;int&#xff09;浮点型&#xff08;float&#xff09;复数&#xff08;complex&#xff09; 3. 字符串类型字符串的定义与使用字符串操作方法 4. 布尔类型布尔值和布尔运算 5. 列…

docker网络详解

1. 网络模式 1.1 网络结构 当安装Docker以后&#xff0c;会自动创建三个网络。可以使用docker network ls命令列出这些网络。 $ docker network ls NETWORK ID NAME DRIVER SCOPE 440aefe8afa3 bridge bridge local aa8d6325580f host host …

chat-glm4,qwen1.5性能对比

modelMMLUC-EvalGSM8KHumanEvalglm-4-9b74.777.184.070.1qwen1.5-7b6174.162.536.0qwen1.5-14b67.678.770.137.8 数据来源是以下两个图。可以看到GLM4非常优秀&#xff0c;qwen应该也快要开源自己的新模型了&#xff0c;希望国内的大模型团队能够继续坚持&#xff0c;持续努力&…

AI框架之Spring AI与Spring Cloud Alibaba AI使用讲解

文章目录 1 AI框架1.1 Spring AI 简介1.2 Spring AI 使用1.2.1 pom.xml1.2.2 可实现的功能 1.3 Spring Cloud Alibaba AI1.4 Spring Cloud Alibaba AI 实践操作1.4.1 pom.xml1.4.2 配置文件1.4.3 对接文本模型1.4.4 文生图模型1.4.5 语音合成模型 1 AI框架 1.1 Spring AI 简介…

NSSCTF-Web题目5

目录 [SWPUCTF 2021 新生赛]error 1、题目 2、知识点 3、思路 [LitCTF 2023]作业管理系统 1、题目 2、知识点 3、思路 [HUBUCTF 2022 新生赛]checkin 1、题目 2、知识点 3、思路 [SWPUCTF 2021 新生赛]error 1、题目 2、知识点 数据库注入、报错注入 3、思路 首先…

vue 文件预览mp4、txt、pptx、xls、xlsx、docx、pdf、html、xml

vue 文件预览 图片、mp4、txt、pptx、xls、xlsx、docx、pdf、html、xml 最近公司要做一个类似电脑文件夹的功能&#xff0c;支持文件夹操作&#xff0c;文件操作,这里就不说文件夹操作了&#xff0c;说说文件预览操作&#xff0c;本人是后端java开发&#xff0c;前端vue&#…

Linux系统安装APITable详细流程与远程访问本地平台数据分析

文章目录 前言1. 部署APITable2. cpolar的安装和注册3. 配置APITable公网访问地址4. 固定APITable公网地址 &#x1f4a1;推荐 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。【点击跳转到网站】 前言 v…

AMD显卡和英伟达显卡哪个好?

显卡是计算机中负责处理图形和视频输出的硬件设备&#xff0c;主要分为两种类型&#xff1a;AMD的A卡和NVIDIA的N卡。那么AMD显卡和英伟达显卡哪个好&#xff1f;怎么选&#xff1f; 答&#xff1a;不能一概而论地说哪个好&#xff0c;因为它们各有优势&#xff0c;选择应基于…

RE_RC4加密

之前做的几道题目&#xff0c;rc4也是经常遇到&#xff0c;今来系统学学&#xff0c;记录一下 对称加密&#xff0c;即加密和解密的密钥可以相互推导&#xff0c;也有的是相同的。 RC4 是以字节流处理每一个字节&#xff0c;而不是 DES 的分组操作。 包含三个参数&#xff1…

探索通信技术的未来:2024中国通信技术和智能装备产业博览会

探索通信技术的未来&#xff1a;2024通信技术产业专场 随着信息技术的飞速发展&#xff0c;通信技术已成为现代社会不可或缺的基础设施。2024年10月11日至13日&#xff0c;青岛将迎来一场通信技术的盛会——2024中国军民两用智能装备与通信技术产业博览会。本次博览会不仅将展…

面试(03)————多线程

目录 一、线程和进程的区别&#xff1f; 二、并行和并发的区别&#xff1f; 三、线程创建的方式有哪些&#xff1f; 3.1、继承Thread类 3.2、实现Runnable接口 3.3、实现Callable接口 3.4、线程池 四、Runnable和Callable的区别&#xff1f; 五、在启动线程的时候&am…

书生·浦语大模型全链路开源体系-笔记作业4

XTuner 微调 LLM:1.8B、多模态、Agent 引自&#xff1a;Tutorial/xtuner/personal_assistant_document.md at camp2 InternLM/Tutorial GitHub 1. XTuner介绍 引自&#xff1a;欢迎来到 XTuner 的中文文档 — XTuner 0.1.18.dev0 文档 1.1. 什么是 XTuner &#xff1f; X…

修改缓存供应商--EhCache

除了我们默认的缓存形式simlpe之外, 我们其实还有许多其他种类的缓存供应 Ehcache就是其中的一种形式 Ehcache在SpringBoot当中的使用: 其实跟我们之前整合第三方的资源是一样的形式 1>导入依赖: <!-- 更换缓存, 将默认使用的 Simple 更换为Ehcache--> <depe…

【go】windows环境设置goos

场景 本地环境&#xff1a;windows 生产环境&#xff1a;linux 现想在本地将go脚本编译为可执行二进制文件&#xff0c;转移至生产中进行运行测试。但go build不生效。 方案&#xff08;修改GOOS&#xff09; cmd打开命令行&#xff0c;执行go env查看本地go环境&#xff0c…