赞
踩
事务是指一组最小的逻辑操作单元,里面有多个操作组成。组成事务的每一部分必须要同时提交成功,如果有一个操作失败,整个操作就回滚。
原子性是指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。
事务必须使数据库从一个一致性状态变换到另外一个一致性状态。
事务的隔离性是多个用户并发访问数据库时,数据库为每一个用户开启的事务,不能被其他事务的操作数据所干扰,多个并发事务之间要相互隔离。
持久性是指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,接下来即使数据库发生故障也不应该对其有任何影响。
转账问题:包括减钱和加钱两个步骤,这是两个不同的SQL语句,默认的这是两个事务,但是这两个事务在提交的过程中,发生了异常,导致减钱成功,加钱失败,该怎么解决?
package com.westo.jdbc3; import com.westo.utils.JDBCUtils; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; public class Demo { public static void main(String[] args) throws SQLException { Connection connection = JDBCUtils.getConnection(); String sql1="update demo2 set money=money-100 where name='张三'"; String sql2="update demo2 set money=money+100 where name='李四'"; PreparedStatement preparedStatement = connection.prepareStatement(sql1); PreparedStatement preparedStatement1 = connection.prepareStatement(sql2); preparedStatement.executeUpdate(); //模拟异常 System.out.println(1/0); preparedStatement1.executeUpdate(); connection.close(); preparedStatement.close(); preparedStatement1.close(); } }
所以,我们要把扣钱和加钱当做一个整体,放在一个事务中统一提交。
package com.westo.jdbc3; import com.westo.utils.JDBCUtils; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.SQLException; public class Demo1 { private static Connection connection; public static void main(String[] args){ try { connection = JDBCUtils.getConnection(); //把数据的事务设置为手动提交 connection.setAutoCommit(false); String sql1="update demo2 set money=money-100 where name='张三'"; String sql2="update demo2 set money=money+100 where name='李四'"; PreparedStatement preparedStatement = connection.prepareStatement(sql1); PreparedStatement preparedStatement1 = connection.prepareStatement(sql2); preparedStatement.executeUpdate(); //模拟异常 System.out.println(1/0); preparedStatement1.executeUpdate(); } catch (Exception e) { try { //遇到异常回滚事务 connection.rollback(); } catch (SQLException e1) { e1.printStackTrace(); } e.printStackTrace(); }finally { try { //不管有没有遇到异常 都提交事务 connection.commit(); } catch (SQLException e) { e.printStackTrace(); } } } }
我们知道隔离性是指一个事务执行的时候最好不要受到其他事务的影响,当我们不考虑隔离性时就会出现读问题。
我们开启两个dos命令窗口来演示,首先要将事务的隔离级别设置为读未提交,因为MySQL默认的隔离级别是避免了这种事情的发生。
set session transaction isolation level read uncommitted;
我们可以看到,1号窗口改变数据之后,2号窗口去查询可以查到,但是当1号回滚之后,2号去查询发现查询不到了。
首先要将事务的隔离级别设置为读已提交
set session transaction isolation level read uncommitted;
我们可以看到,事务提交之前和之后,数据查询的结果是不同的,这就是不可重复度。
read uncommitted>read committed>repeatable read>serializable
四种隔离级别的安全性
read uncommitted<read committed<repeatable read<serializable
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。