JDBC-3(Transcation) ****
3.1 异常的使用说明
- 在工具类中(JDBCUtils)的方法最好声明异常(throws),以便后续实现类中去捕获这些异常。
- 工具类中捕获异常通常没有意义 eg:实现类中connection建立过程出现异常,若工具类中使用try-catch则实现类处理异常后后续仍可继续执行,此时后续执行便无意义。
- 在实现类中去捕获并处理异常,且可使用fially去将开启的资源(connection, preparedStatement, I/O)关闭
3.2 事务简介
3.2.1 事务&事务处理
- 事务:一组逻辑操作单元,使数据从一种状态转变为另一种状态
- 事务处理(事务操作):当在一个事务中执行多个操作时,要么所有的事务都被提交(commit),那么这些修改就永久地保存下来;要么数据库管理系统将放弃所作的所有修改,整个事务回滚(rollback)到最初状态。
3.2.2 JDBC事务处理
- 事务一旦提交就不可回滚
- 当一个连接对象被创建时,默认情况下是自动提交事务
AutoCommit(true):每次执行一个 SQL 语句时,如果执行成功,就会向数据库自动提交,而不能回滚。 - 关闭数据库连接,数据就会自动的提交。 如果多个操作,每个操作使用的是自己单独的连接,则无法保证事务。即同一个事务的多个操作必须在同一个连接下。
- 当一个连接对象被创建时,默认情况下是自动提交事务
JDBC的事务处理:
- 调用Connection对象的setAutoCommit(false)取消自动提交事务
- 所有SQL语句执行成功后调用commit()提交事务
- 出现异常时在catch里调用rollback()回滚事务
- 若此时 Connection 没有被关闭,还可能被重复使用,则需要恢复其自动提交状态 setAutoCommit(true)。尤其是在使用数据库连接池技术时,执行close()方法前,建议恢复自动提交状态。
点击查看实例
@Test//updateRollBack()实现sql语句执行
public void testUpdate(){
Connection connection = null;
try {
//1.取消数据的自动提交
connection = JDBCUtils.getConnection();
connection.setAutoCommit(false);
String sql1 = "UPDATE account\n" +
"SET balance = balance - 100\n" +
"WHERE id = ?;";
String sql2 = "UPDATE account\n" +
"SET balance = balance + 100\n" +
"WHERE id = ?;";
updateRollBack(connection, sql1, 1);
//System.out.println(10 / 0);
updateRollBack(connection, sql2, 2);
//2.提交数据
connection.commit();
System.out.println("转账成功!");
} catch (Exception e){
e.printStackTrace();
System.out.println("转账失败!");
try {
//3.出现异常数据回滚
connection.rollback();
} catch (Exception e2){
e2.printStackTrace();
}
} finally {
JDBCUtils.closeResource(connection, null);
}
}
3.2.3 事务的四大属性(ACID)
- 原子性Atomicity: 过程不可分割
- 指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。
- 一致性Consistency: 操作前与操作后总量不变
- 使数据库从一个一致性状态变换到另外一个一致性状态。
- 隔离性Isolation: 多事务操作时,互相不会产生影响
- 指一个事务的执行不能被其他事务干扰,即一个事务内部的操作及使用的数据对并发的其他事务是隔离的,并发执行的各个事务之间不能互相干扰。
- 持久性Durability: 事务提交后表中数据发生变化
- 一个事务一旦被提交,它对数据库中数据的改变就是永久性的,接下来的其他操作和数据库故障不应该对其有任何影响。
3.2.4 数据库的并发问题&隔离级别
1-并发问题
- 脏读:读到了已更新而未提交的数据
- 对于两个事务 T1, T2, T1 读取了已经被 T2 更新但还没有被提交的字段。之后, 若 T2 回滚, T1读取的内容就是临时且无效的。
- 不可重复读: 读到了已更新(update)已提交字段
- 对于两个事务T1, T2, T1 读取了一个字段, 然后 T2 更新了该字段。之后, T1再次读取同一个字段, 值就不同了。
- 幻读: 读到了已插入(insert)已提交字段
- 对于两个事务T1, T2, T1 从一个表中读取了一个字段, 然后 T2 在该表中插入了一些新的行。之后, 如果 T1 再次读取同一个表, 就会多出几行。
2-隔离级别
- 一个事务与其他事务隔离的程度称为隔离级别,隔离级别越高,数据一致性越好,但并发性越差。
- READ UNCOMMITTED(读未提交):允许事务读取其他食物未被提交的变更。允许脏读的出现
- READ COMMITTED(读已提交):只允许事务读取已被其他事物提交的变更。不允许脏读,允许可重复读
- REPEATABLE READ(可重复读):确保事务可以多此从一个字段中读取相同的值,事务持续期间禁止任何其他事务对这个字段进行更新。 避免脏读和可重复读
- SERIALIZABLE(串行化):确保事务可以从一个表中读取相同的行,事务持续期间禁止其他事务对该表进行CUD
3-MySQL中设置隔离级别
- 查看当前隔离级别:
SELECT @@tx_isolation; - 设置当前MySQL连接隔离级别:
set transaction isolation level read committed; - 设置数据库系统的全局隔离级别 :
set global transaction isolation level read committed; - 补充
- 创建mysql数据库用户:
create user tom identified by 'abc123'; - 授予权限:
- 授予通过网络方式登录的tom用户,对所有库所有表的全部权限,密码设为root
grant all privileges on *.* to tom@'%' identified by 'root'; - 给tom用户使用本地命令行方式,授予user_db这个库下的所有表的插删改查的权限。
grant select,insert,delete,update on user_db.* to tom@localhost identified by 'abc123';
- 授予通过网络方式登录的tom用户,对所有库所有表的全部权限,密码设为root
- 创建mysql数据库用户:
4-Java中设置隔离级别
- 设置隔离级别:
connection.setTranscationIsolation(Connection.XXX); - 获取隔离级别:
connection.getTransactionIsolation();
点击查看隔离级别测试代码
@Test
public void select() throws Exception{
Connection connection = JDBCUtils.getConnection();
//获取当前隔离级别
System.out.println(connection.getTransactionIsolation());
//设置隔离级别
connection.setTransactionIsolation(Connection.TRANSACTION_READ_UNCOMMITTED);
connection.setAutoCommit(false);
String sql = "select * from account;";
List<Account> list = AllSelect.getInstance(connection, Account.class, sql);
for(int i = 0; i < list.size(); i++){
System.out.println(list.get(i));
}
}
@Test
public void update() throws Exception{
Connection connection = JDBCUtils.getConnection();
String sql = "update account set balance = balance + 100 where id = ?";
connection.setAutoCommit(false);
PreparedStatement preparedStatement = connection.prepareStatement(sql);
preparedStatement.setObject(1, 3);
preparedStatement.executeUpdate();
//线程休眠15s
Thread.sleep(15000);
//回滚测试脏读
connection.rollback();
}
JDBC-3(Transcation) ****的更多相关文章
- Java数据库连接技术——JDBC
大家好,今天我们学习了Java如何连接数据库.之前学过.net语言的数据库操作,感觉就是一通百通,大同小异. JDBC是Java数据库连接技术的简称,提供连接各种常用数据库的能力. JDBC API ...
- 玩转spring boot——结合AngularJs和JDBC
参考官方例子:http://spring.io/guides/gs/relational-data-access/ 一.项目准备 在建立mysql数据库后新建表“t_order” ; -- ----- ...
- [原创]java使用JDBC向MySQL数据库批次插入10W条数据测试效率
使用JDBC连接MySQL数据库进行数据插入的时候,特别是大批量数据连续插入(100000),如何提高效率呢?在JDBC编程接口中Statement 有两个方法特别值得注意:通过使用addBatch( ...
- JDBC MySQL 多表关联查询查询
public static void main(String[] args) throws Exception{ Class.forName("com.mysql.jdbc.Driver&q ...
- JDBC增加删除修改
一.配置程序--让我们程序能找到数据库的驱动jar包 1.把.jar文件复制到项目中去,整合的时候方便. 2.在eclipse项目右击"构建路径"--"配置构建路径&qu ...
- JDBC简介
jdbc连接数据库的四个对象 DriverManager 驱动类 DriverManager.registerDriver(new com.mysql.jdbc.Driver());不建议使用 ...
- JDBC Tutorials: Commit or Rollback transaction in finally block
http://skeletoncoder.blogspot.com/2006/10/jdbc-tutorials-commit-or-rollback.html JDBC Tutorials: Com ...
- FineReport如何用JDBC连接阿里云ADS数据库
在使用FineReport连接阿里云的ADS(AnalyticDB)数据库,很多时候在测试连接时就失败了.此时,该如何连接ADS数据库呢? 我们只需要手动将连接ads数据库需要使用到的jar放置到%F ...
- JDBC基础
今天看了看JDBC(Java DataBase Connectivity)总结一下 关于JDBC 加载JDBC驱动 建立数据库连接 创建一个Statement或者PreparedStatement 获 ...
- Spring学习记录(十四)---JDBC基本操作
先看一些定义: 在Spring JDBC模块中,所有的类可以被分到四个单独的包:1.core即核心包,它包含了JDBC的核心功能.此包内有很多重要的类,包括:JdbcTemplate类.SimpleJ ...
随机推荐
- C#/VB.NET 将PDF转为OFD
OFD,全称Open Fixed-layout Document ,是一种可存储.读取以及编辑的国家标准版式的电子文档格式,属于中国的一种自主文件格式,在安全性上有可靠保证.为突破国外技术在我们软硬件 ...
- 题解 P4336 [SHOI2016]黑暗前的幻想乡
题解 前置芝士 :矩阵树定理 本题是一道计数题,有两个要求: 建造的公路构成一颗生成树 每条公路由不同的公司建造,每条公路与一个公司一一映射 那么看到这两个要求后,我们很容易想到第一个条件用矩阵树定理 ...
- FirstGradle
一.导入依赖 二.build.gradle 整合SpringBoot plugins { id 'java' } group 'com.qiang' version '1.0.0-SNAPSHOT' ...
- MongoDB用户创建
MongoDB采用基于角色的访问控制(RBAC)来确定用户的访问. 授予用户一个或多个角色,确定用户对MongoDB资源的访问权限和用户可以执行哪些操作. 用户应该只有最小权限集才能确保最小权限的系统 ...
- java中jre\bin目录和jdk\bin目录下的工具功能介绍
转自:https://blog.csdn.net/eclipse_yin/article/details/51051096 jre/bin目录下面工具说明 javac:Java编译器,将Java源代码 ...
- WPF---依赖属性(二)
一.概要 我们将会通过一个简单的综合例子来阐述下依赖属性的变化. 场景:我们在一个文本框中输入一个数字,然后对应的panel中会出现对应的椭圆,椭圆的个数与输入的文本相同. 我们在MainWindow ...
- [SWMM]软件启动不了,出现 “ RPC服务器不可用 ” 错误
[问题]打开SWMM5.1软件时,初选"RPC服务器不可用"的错误 [解决]计算机管理--服务 设置Print Spooler服务状态为启动,并设置为自启动.
- python turtle的使用
turtle.pendown() # 放下画笔 turtle.penup() # 抬起画笔 turtle.pensize(int) # 设置画笔宽度,值为整数型 turtle.forward(f ...
- 【CSS简介、基础选择器、字体属性、文本属性、引入方式】前端小抄(2) - Pink老师自学笔记
[CSS简介.基础选择器.字体属性.文本属性.引入方式]前端小抄(2) 本学习笔记是个人对 Pink 老师课程的总结归纳,转载请注明出处! 一.CSS简介 CSS 的主要使用场景就是布局网页,美化页面 ...
- spring之自定义注解
java.lang.annotation提供了四种元注解,专门注解其他的注解: @Target –注解用于什么地方,默认值为任何元素,表示该注解用于什么地方.可用的ElementType指定参数 ,也 ...