工具类解决事务和过滤器解决事务

事务的四大特性ACID

原子性:强调事务的不可分割.多条语句要么都成功,要么都失败。
一致性:强调的是事务的执行的前后,数据要保持一致
隔离性:并发访问数据库时,一个事务的执行不应该受到其他事务的干扰.
持久性:一个事务被提交之后。它对数据库中数据的改变是持久的,即使数据库发生故障也不应该对其有任何影响。

事务的隔离级别

READ-UNCOMMITTED(读取未提交): 最低的隔离级别,允许读取尚未提交的数据变更,可能会导致脏读、幻读或不可重复读

READ-COMMITTED(读取已提交): 允许读取并发事务已经提交的数据,可以阻止脏读,但是幻读或不可重复读 仍有可能发生

REPEATABLE-READ(可重复读): 对同一字段的多次读取结果都是一致的,除非数据是被本身事务自己所修 改,可以阻止脏读和不可重复读,但幻读仍有可能发生。

SERIALIZABLE(可串行化): 最高的隔离级别,完全服从ACID的隔离级别。所有的事务依次逐个执行,这样事务 之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读以及幻读。
在这里插入图片描述

JdbcUtils 事务工具类

public class JdbcUtils {//准备Properties集合private static Properties config=new Properties();//连接池private static DataSource dataSource;//当前线程private static ThreadLocal<Connection> tl=new ThreadLocal<>();static {try {//1.读取properties文件的内容InputStream in = JdbcUtils.class.getClassLoader().getResourceAsStream("datasource.properties");config.load(in);//2.创建连接池DruidDataSource druidDataSource=new DruidDataSource();druidDataSource.setDriverClassName(config.getProperty("jdbc.driverClassName"));druidDataSource.setUrl(config.getProperty("jdbc.url"));druidDataSource.setUsername(config.getProperty("jdbc.username"));druidDataSource.setPassword(config.getProperty("jdbc.password"));dataSource=druidDataSource;}catch (Exception e){e.printStackTrace();throw new RuntimeException("创建数据库连接池失败!");}}/*** 获取链接* @return*/public static Connection getConnection(){try {//1.先从当前线程获取链接Connection connection = tl.get();//2.如果当前线程连接为空,代表第一次调用if(connection==null){//2.1获取一个新的连接connection=dataSource.getConnection();//2.2 绑定连接到当前线程tl.set(connection);}return connection;} catch (Exception e) {e.printStackTrace();throw new RuntimeException("获取连接失败!");}}/*** 开启事务*/public static void startTransaction(){try {//1.获取链接Connection connection = getConnection();//2.开启事务connection.setAutoCommit(false);}catch (Exception e){e.printStackTrace();throw new RuntimeException("开启事务");}}/*** 提交事务*/public static void commitTransaction(){try {//1.获取链接Connection connection = tl.get();//2.如果当前线程有连接才提交事务if (connection != null) {connection.commit();}}catch (Exception e){e.printStackTrace();throw new RuntimeException("提交事务失败!");}}/*** 回滚事务*/public static void rollback(){try {//1.获取链接Connection connection = tl.get();//2.如果当前线程有连接才回滚if (connection != null) {connection.rollback();connection.commit();}}catch (Exception e){e.printStackTrace();throw new RuntimeException("回滚事务失败!");}}/*** 关闭链接,移除当前线程绑定的连接*/public static void closeTransaction(){try {//1.获取链接Connection connection = tl.get();if (connection != null) {//移除当前线程绑定的连接tl.remove();//关闭链接connection.close();}}catch (Exception e){e.printStackTrace();throw new RuntimeException("关闭链接失败!");}}/*** 关闭资源* @param st* @param rs*/public static void close(Statement st, ResultSet rs){if(rs!=null){try {rs.close();} catch (SQLException e) {e.printStackTrace();}rs=null;}if(st!=null){try {st.close();} catch (SQLException e) {e.printStackTrace();}st=null;}}}

dao

public interface AccountDao {/*** 根据id查询账户信息* @param id* @return*/Account findAccountById(Integer id);/*** 修改账户信息* @param account*/void updateAccount(Account account);}
public class AccountDaoImpl implements AccountDao {@Overridepublic Account findAccountById(Integer id) {PreparedStatement st=null;ResultSet rs=null;try{Connection conn=JdbcUtils.getConnection();//3.准备sqlString sql="select id,name,money from account where id=?";st=conn.prepareStatement(sql);//4.设置参数st.setInt(1,id);//5.执行sqlrs=st.executeQuery();//6.解析结果Account account=null;if(rs.next()){account=new Account();account.setId(id);account.setName(rs.getString("name"));account.setMoney(rs.getDouble("money"));}return account;}catch (Exception e){e.printStackTrace();throw new RuntimeException("查询失败!");}finally {JdbcUtils.close(st,rs);}}@Overridepublic void updateAccount(Account account) {PreparedStatement st=null;ResultSet rs=null;try{Connection conn=JdbcUtils.getConnection();//3.准备sqlString sql="update account set money=?,name=? where id=?";st=conn.prepareStatement(sql);//4.设置参数st.setDouble(1,account.getMoney());st.setString(2,account.getName());st.setInt(3,account.getId());//5.执行sqlst.executeUpdate();}catch (Exception e){e.printStackTrace();throw new RuntimeException("修改失败!");}finally {JdbcUtils.close(st,rs);}}
}

service

public class AccountServiceImpl implements AccountService {private AccountDao accountDao=new AccountDaoImpl();@Overridepublic void transfer(Integer sourceId, Integer targetId, Double money) {//1.查询源账户信息Account sAccount = accountDao.findAccountById(sourceId);//2.查询目标账户信息Account tAccount = accountDao.findAccountById(targetId);//3.转账的业务if (sAccount.getMoney() < money) {throw new RuntimeException("余额不足!");}//4.源账户扣钱sAccount.setMoney(sAccount.getMoney() - money);accountDao.updateAccount(sAccount);//int x = 1 / 0;//5.目标账户加钱tAccount.setMoney(tAccount.getMoney() + money);accountDao.updateAccount(tAccount);}}

servlet

@WebServlet("/accountServlet")
public class AccountServlet extends HttpServlet {private AccountService accountService=new AccountServiceImpl();@Overrideprotected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {try {//1.开启事务JdbcUtils.startTransaction();//2.执行业务AccountService accountService = new AccountServiceImpl();accountService.transfer(1, 2, 3.0);//3.提交事务JdbcUtils.commitTransaction();}catch (Exception e){e.printStackTrace();//回滚事务JdbcUtils.rollback();}finally {//关闭资源,移除当前线程绑定的连接JdbcUtils.closeTransaction();}}
}

过滤器解决事务

@WebFilter("/*")
public class TransactionFilter implements Filter {@Overridepublic void init(FilterConfig filterConfig) throws ServletException {}@Overridepublic void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {try{//1.开启事务JdbcUtils.startTransaction();//2.执行业务  放行chain.doFilter(req,res);//3.提交事务JdbcUtils.commitTransaction();}catch (Exception e){e.printStackTrace();//回滚事务JdbcUtils.rollback();}finally {//关闭资源,移除当前线程绑定的连接JdbcUtils.closeTransaction();}}@Overridepublic void destroy() {}
}

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

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

相关文章

测试:ollama加载羊驼版本llama-3中文大模型

找了一个晚上各种模型&#xff0c;像极了当初找各种操作系统的镜像&#xff0c;雨林木风&#xff0c;深蓝、老毛桃…… 主要是官方的默认7B版本回答好多英文&#xff0c;而且回复的很慢&#xff0c;所以我是在ollama上搜索"chinese"找到了这个羊驼版本的&#xff0c…

使用javacv对摄像头视频转码并实现播放

要实现Java接受RTSP流解码&#xff0c;并推送给前端实现播放实时流&#xff0c;可以使用一些流媒体处理库&#xff0c;比如JavaCV或者FFmpeg等。以下是一个简单的示例代码&#xff1a; 1.控制层方面的 根据视频rtsp流链接打开转换&#xff0c;通过响应写出流到前台使用flvjs播…

go语言初学04

Go 语言近年来发展迅速&#xff0c;并且出现了许多优秀的开发框架和组件来支持各种不同的开发需求。以下是一些常用的 Go 语言开发框架和组件&#xff1a; Web 框架 Gin&#xff1a; URL: Gin简单、高效、易用&#xff0c;适合构建高性能的 Web 应用。 Echo&#xff1a; URL: …

crossover软件是干什么的 crossover软件安装使用教程 crossover软件如何使用

CrossOver 以其出色的跨平台兼容性&#xff0c;让用户在Mac设备上轻松运行各种Windows软件&#xff0c;无需复杂的设置或额外的配置&#xff0c;支持多种语言&#xff0c;满足不同国家和地区用户的需求。 CrossOver 软件是干嘛的 使用CrossOver 不必购买Windows 授权&#xf…

Winform ListView 嵌入组合框、布尔、图片等复杂控件

一、Winform ListView 显示复杂控件示例 以下展示了两种实现思路方案。最后修改日期 2024-05 surfsky 1.1 方案一&#xff1a;ListView 结合组合框进行模拟编辑 基本思路 在界面上放置一个lisview和一个combobox&#xff0c;combobox平时是隐藏的。点击listview&#xff0c…

ArrayList源码讲解

ArrayList 底层采用的是数组队列&#xff0c;相当于动态数组。 ArrayList内部使用一个可重新分配的Object数组来存储元素&#xff0c;这个数组会随着元素的添加自动增长以容纳更多的元素&#xff0c;这就是所谓的“动态数组”。 1.实现了RandomAccess接口&#xff0c;可以随机…

rust嵌入式开发之总结

我们用rust开发的新版产品刚刚交付&#xff0c;已经在海上安装测试完毕并顺利投产。终于松了口气&#xff0c;同时也有时间和精力来做个全面的总结了。 这个产品&#xff0c;目前差不多有三版&#xff1a; 第一个版本是用crt-thread写的&#xff0c;投产后出了一个内存泄露的…

521源码-源码论坛-宝塔面板操作日志是存放在哪里的? 如何删除部分日志记录

我们帮别人搭建或者登录了&#xff08;不是自己权属的宝塔面板&#xff09;&#xff0c;会留下登录及操作的日志&#xff0c;我们不想留下这些操作日志&#xff0c;可以通过下面的方法处理掉&#xff0c;以达到无痕迹访问操作的目的&#xff1a; 如图所示的面板操作日志&#…

Python-3.12.0文档解读-内置函数sum()详细说明+记忆策略+常用场景+巧妙用法+综合技巧

一个认为一切根源都是“自己不够强”的INTJ 个人主页&#xff1a;用哲学编程-CSDN博客专栏&#xff1a;每日一题——举一反三Python编程学习Python内置函数 Python-3.12.0文档解读 目录 详细说明 sum(iterable, /, start0) 参数&#xff1a; 返回值&#xff1a; 注意事…

骑砍2霸主MOD开发(10)-游戏大地图(MapScene)制作

一.MapScene中初始化NavMeshFaceID与TerrainType public TerrainType GetFaceTerrainType(PathFaceRecord navMeshFace){switch (navMeshFace.FaceGroupIndex){case 1:return TerrainType.Plain;case 2:return TerrainType.Desert;case 3:return TerrainType.Snow;case 4:retur…

算法优化:空间与时间复杂度的权衡

引言 在软件开发中&#xff0c;算法的性能至关重要。算法的性能通常通过其时间复杂度和空间复杂度来衡量。时间复杂度指的是算法执行时间与输入规模的关系&#xff0c;而空间复杂度则关注算法执行过程中所占用的存储空间。本文将探讨如何权衡这两者&#xff0c;以实现算法的最…

排序方法大汇总

以下所有排序方法均以排升序为例 一.插入排序 1.直接插入排序 1>方法介绍&#xff1a;假定前n个数据有序&#xff0c;将第n1个数据依次与前n个数据相比&#xff0c;若比第i个数据小且比第i-1个数据大则插入两者之间 2>时间复杂度&#xff1a;O(N^2) 空间复杂度&#…

【JS】对象转变成数组

1、Object.keys() 方法&#xff1a; 将对象的键转换为数组 const a { name: aa,age: 18 }; const arr Object.keys(a); console.log(arr); // 输出 [name, age] 2、Object.values() 方法&#xff1a; 将对象的值转换为数组 const a { name: aa,age: 18 }; const arr Obj…

BUUCTF中的密码题目解密

BUUCTF 1.MD5 题目名称就是MD5&#xff0c;这个题目肯定和md5密码有关&#xff0c;下载题目&#xff0c;打开后发现这确实是一个md5加密的密文 Md5在线解密网站&#xff1a;md5在线解密破解,md5解密加密 经过MD5在线解密网站解密后&#xff0c;获取到flag为&#xff1a;flag{…

域名主机服务器配置失败的原因和解决方法

域名主机服务器配置失败的原因可能涉及多个方面&#xff0c;包括域名设置、DNS配置、服务器设置、网络问题等。以下是一些常见的原因和相应的解决方案&#xff1a; 1. DNS配置错误 原因&#xff1a; 域名解析错误&#xff1a;域名没有正确指向服务器的IP地址。 DNS记录未更新&a…

网络编程TCP

White graces&#xff1a;个人主页 &#x1f649;专栏推荐:Java入门知识&#x1f649; &#x1f649; 内容推荐:Java网络编程(下)&#x1f649; &#x1f439;今日诗词: 壮士当唱大风哥, 宵小之徒能几何&#xff1f;&#x1f439; ⛳️点赞 ☀️收藏⭐️关注&#x1f4ac;卑微…

CentOS7单用户模式,救援模式操作记录

CentOS7单用户模式&#xff0c;救援模式操作记录 1. 单用户模式 单用户模式进入不需要密码&#xff0c;无网络连接&#xff0c;拥有root权限&#xff0c;禁止远程登陆。一般用于用于系统维护&#xff0c;例如忘记root密码后可以通过进入单用户模式进行重置。 开机启动&#…

数据结构 实验 1

题目一&#xff1a;用线性表实现文具店的货品管理问题 问题描述&#xff1a;在文具店的日常经营过程中&#xff0c;存在对各种文具的管理问题。当库存文具不足或缺货时&#xff0c;需要进货。日常销售时需要出库。当盘点货物时&#xff0c;需要查询货物信息。请根据这些要求编…

使用低代码系统的意义与价值主要体现在哪里?

使用低代码系统的意义与价值主要体现在以下几个方面&#xff0c;这些观点基于驰骋低代码设计者的专业洞察和行业经验&#xff1a; 快速原型创建&#xff1a; 低代码平台通过提供图形化界面和预构建的模块&#xff0c;极大地加速了系统原型的创建过程。这意味着企业能够更快地验…

60 关于 SegmentFault 的一些场景 (1)

前言 呵呵 此问题主要是来自于 帖子 月经结贴 -- 《Segmentation Fault in Linux》 这里主要也是 结合了作者的相关 case, 来做的一些 调试分享 当然 很多的情况还是 蛮有意思 本文主要问题如下 1. 访问可执行文件中的 只读数据 2. 访问不存在的虚拟地址 3. 访问内核地址…