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

事务的四大特性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…

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…

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; 注意事…

排序方法大汇总

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

BUUCTF中的密码题目解密

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

网络编程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. 访问内核地址…

嵌入式工程师人生提质的十大成长型思维分享

大家好,作为一名嵌入式开发者,很多时候,需要考虑个人未来的发展,人生旅途复杂多变,时常面临各种各样的挑战。如何在这个复杂多变的社会中稳步向前,不断成长,成为每个人都应该思考的问题。实际上,思维方式的差异决定我们应对挑战的能力与成长的速度。 第一:寻找自我坐…

HNCTF2022 REVERSE

[HNCTF 2022 WEEK2]esy_flower 简单花指令 Nop掉 然后整段u c p然后就反汇编 可能反编译的不太对&#xff0c;&#xff0c;看了别人的wp就是ida反编译的有问题 #include<stdio.h> #include<string.h> int main() {int i,j;char ch[]"c~scvdzKCEoDEZ[^roDICU…

微软远程连接工具:Microsoft Remote Desktop for Mac 中文版

Microsoft Remote Desktop 是一款由微软开发的远程桌面连接软件&#xff0c;它允许用户从远程地点连接到远程计算机或虚拟机&#xff0c;并在远程计算机上使用桌面应用程序和文件。 下载地址&#xff1a;https://www.macz.com/mac/5458.html?idOTI2NjQ5Jl8mMjcuMTg2LjEyNi4yMz…

C++进阶之AVL树+模拟实现

目录 目录 一、AVL树的基本概念 1.1 基本概念 二、AVL树的模拟实现 2.1 AVL树节点的定义 2.2 插入操作 2.3 旋转操作 2.4 具体实现 一、AVL树的基本概念 1.1 基本概念 二叉搜索树虽可以缩短查找的效率&#xff0c;但如果数据有序或接近有序二叉搜索树将退化为单支树&…

山东大学软件学院项目实训-创新实训-基于大模型的旅游平台(二十四)- 微服务(4)

目录 8. http客户端Feign 8.1 feign远程调用 8.2 feign自定义配置 8.3 feign性能优化 8.4 feign最佳实践 8. http客户端Feign 8.1 feign远程调用 RestTemplate存在的问题 &#xff1a; 代码可读性差 参数复杂URL难以维护 Feign是声明式的http客户端 使用步骤 &#xf…

飞书API(11):阿里云MaxCompute分区表入库

一、引入 前面入库阿里云 MaxCompute 的数据都是读取之后直接写入&#xff0c;保留数据最新的状态&#xff0c;如果我要保留历史的状态&#xff0c;怎么办呢&#xff1f;MaxCompute 表有一个分区功能&#xff0c;可以自行定义分区。我们可以使用 MaxCompute 表的分区功能&…

远程自动锁定平面

目录 Ubuntu 系统上 方法一&#xff1a;使用 SSH 重新连接 方法二&#xff1a;解锁当前会话 方法三&#xff1a;通过 SSH 解锁会话 方法四&#xff1a;禁用自动锁屏&#xff08;如果合适&#xff09; windows系统 方法三&#xff1a;修改组策略设置 Ubuntu 系统上 远程…

重生之我要精通JAVA--第七周笔记

文章目录 IO流字符流字符流原理解析flush和close方法 文件拷贝代码文件加密解密修改文件中的数据 缓冲流字节缓冲流字符缓冲流例题 转换流序列化流序列化流/对象操作输出流 反序列化流序列化流/反序列化流的细节汇总打印流字节打印流字符打印流 解压缩流压缩流Commons-io常见方…