Tom给jerry汇款500元

1、如果多个操作,每个操作使用的是自己的单独的连接,则无法保证事务

2、具体步骤:
   1>事务开始操作前,开始事务:取消Connection的默认提交行为

2>如果事务的操作都成功,都提交事务
   3>回滚事务:若出现异常,则在catch块中回滚事务;

3、模板 try {
  
   //开始事务:取消默认提交(默认自动提交) con.setAutoCommit(false);
  
   //...
  
   
  //提交事务 con.commit();
  
  } catch (Exception e) {
  
  //...
  
 //如果出现异常就回滚事

try {

con.rollback();

} catch (SQLException e1) {
            e1.printStackTrace();

}

}

@Test
 public void testTransaction() {
  
    Connection con = null;
    try {

    con = JDBCTools.getConnection();
       System.out.println("默认事务:" + con.getAutoCommit());
       // 开始事务:取消默认提交(默认自动提交)
       con.setAutoCommit(false);
       String sql = "update users set balance=balance-500 where id=1";
       update(con, sql);
       int i = 10 / 0;
       sql = "update users set balance=balance+500 where id=2";
       update(con, sql);
       // 提交事务
       con.commit();

   } catch (Exception e) {
       // 如果出现异常就回滚事务
       try {
          con.rollback();
       } catch (SQLException e1) {
          e1.printStackTrace();
       }
    } finally {
       JDBCTools.release(null, null, con);
    }

}

设置事务的隔离级别

@Test
 public void testTransactionIsolationRead() {
  
  String sql = "select balance from users where id=1";
  Integer balance=getForValue(sql);
  System.out.println("balance="+balance);
 }

/*
  * 测试事务的隔离级别 在JDBC程序中可以通过Connection的setTransactionIsolation()方法 来设置事务的隔离级别
  */
 @Test
 public void testTransactionIsolationUpdate() {

Connection con = null;

try {

con = JDBCTools.getConnection();
   // 开始事务:取消默认提交(默认自动提交)
   con.setAutoCommit(false);
   String sql = "update users set balance=balance-500 where id=1";
   update(con,sql);
   con.commit();

} catch (Exception e) {
   e.printStackTrace();
  } finally {

}
 }

// 返回某个对象的属性值
 @SuppressWarnings("unchecked")
 public <E> E getForValue(String sql, Object... args) {

// 1、得到结果集:该结果集应该只有一行且只有一列
  Connection con = null;
  PreparedStatement ps = null;
  ResultSet rs = null;
  try {

con = JDBCTools.getConnection();
   //con.setTransactionIsolation(Connection.TRANSACTION_READ_UNCOMMITTED);
   con.setTransactionIsolation(Connection.TRANSACTION_READ_COMMITTED);
   ps = con.prepareStatement(sql);
   for (int i = 0; i < args.length; i++) {
    ps.setObject(i + 1, args[i]);
   }
   rs = ps.executeQuery();

if (rs.next()) {
    return (E) rs.getObject(1);
   }

} catch (Exception e) {
   e.printStackTrace();
  } finally {
   JDBCTools.release(rs, ps, con);
  }

// 2、取得结果集的
  return null;
 }

更新后的JDBCTools.java工具类如下:

import java.io.InputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Properties;

/*
 * JDBC工具类
 * */
public class JDBCTools {

  // 提交事务
   public static void commit(Connection con) {

     if (con != null) {
         try {
            con.commit();
         } catch (SQLException e) {
            e.printStackTrace();
         }
      }
   }

  //回滚事务
   public static void rollback(Connection con) {

    if (con != null) {
         try {
            con.rollback();
           } catch (SQLException e) {
              e.printStackTrace();
           }
        }
   }
 
   //开始事务
   public static void beginTx(Connection con) {

     if (con != null) {
         try {
            con.setAutoCommit(false);
         } catch (SQLException e) {
            e.printStackTrace();
         }
      }
   }

   /*
    * 执行SQL的方法 insert,update,delete
    */
   public static void update(String sql, Object... args) {

     Connection conn = null;
      PreparedStatement ps = null;

     try {
         /*
          * 1、获取Connection连接 2、获取Statement 3、SQL语句 4、关闭数据库连接
          *
          */
         conn = getConnection();
         ps = conn.prepareStatement(sql);

      for (int i = 0; i < args.length; i++) {
            ps.setObject(i + 1, args[i]);
         }

        ps.executeUpdate();

    } catch (Exception e) {
         e.printStackTrace();
      } finally {
         release(null, ps, conn);
      }
   }

  public static Connection getConnection() throws Exception {

    String driverClass = null;
      String jdbcUrl = null;
      String user = null;
      String password = null;

    // 读取类路径下的jdbc.properties文件
      InputStream in = JDBCTools.class.getClassLoader().getResourceAsStream("jdbc.properties");
      Properties properties = new Properties();
      properties.load(in);

    driverClass = properties.getProperty("driver");
      jdbcUrl = properties.getProperty("jdbcUrl");
      user = properties.getProperty("user");
      password = properties.getProperty("password");
      // 加载数据库驱动程序
      Class.forName(driverClass);
      // 通过DriverManager的getConnection()方法获取数据库连接
      Connection connection = DriverManager.getConnection(jdbcUrl, user, password);
      return connection;

  }

  public static void release(ResultSet rs, Statement st, Connection conn) {

    if (rs != null) {
         try {
            rs.close();
         } catch (SQLException e) {
            e.printStackTrace();
         }
      }

     if (st != null) {
         try {
            st.close();
         } catch (SQLException e) {
            e.printStackTrace();
         }
      }

     if (conn != null) {
         try {
            conn.close();
         } catch (SQLException e) {
            e.printStackTrace();
         }
      }
   }
}

<十一>JDBC_事务的处理+隔离的更多相关文章

  1. 深入解析Mysql中事务的四大隔离级别及其所解决的读现象

    本文详细介绍四种事务隔离级别,并通过举例的方式说明不同的级别能解决什么样的读现象.并且介绍了在关系型数据库中不同的隔离级别的实现原理. 在DBMS中,事务保证了一个操作序列可以全部都执行或者全部都不执 ...

  2. Spring事务传播机制&隔离级别

    一.Propagation (事务的传播属性) Propagation : key属性确定代理应该给哪个方法增加事务行为.这样的属性最重要的部份是传播行为.有以下选项可供使用:PROPAGATION_ ...

  3. 数据库事务中的隔离级别和锁+spring Transactional注解

    数据库事务中的隔离级别和锁 数据库事务在后端开发中占非常重要的地位,如何确保数据读取的正确性.安全性也是我们需要研究的问题.ACID首先总结一下数据库事务正确执行的四个要素(ACID): 原子性(At ...

  4. Spring事务传播、隔离等级

    事务传播 PROPAGATION_REQUIRED 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中.这是最常见的选择. PROPAGATION_SUPPORTS 支持当前事 ...

  5. mysql事务之间的隔离级别

    事务间未做隔离,会引起下面这些问题. 1.脏读:一个事务可读到另外一个尚未commit的事务中的数据. 2.不可重复读:在一个事务中,读取同一个数据 a,b,按顺序读取,在读a  b 之间,另外一个事 ...

  6. MySQL 笔记整理(8.b) --事务到底是隔离还是不隔离的?

    笔记记录自林晓斌(丁奇)老师的<MySQL实战45讲> (本篇内图片均来自丁奇老师的讲解,如有侵权,请联系我删除) 8.a) --事务到底是隔离还是不隔离的? 本周工作较忙,加上懒惰,拖更 ...

  7. MySQL 笔记整理(8.a) --事务到底是隔离还是不隔离的?

    笔记记录自林晓斌(丁奇)老师的<MySQL实战45讲> 8.a) --事务到底是隔离还是不隔离的? 这部分内容不太容易理解,笔者也是进行了多次阅读.因此引用原文: 之前有提到过,如果是在可 ...

  8. 数据库事务的四大特性以及事务的隔离级别-与-Spring事务传播机制&隔离级别

    数据库事务的四大特性以及事务的隔离级别   本篇讲诉数据库中事务的四大特性(ACID),并且将会详细地说明事务的隔离级别. 如果一个数据库声称支持事务的操作,那么该数据库必须要具备以下四个特性: ⑴ ...

  9. 框架源码系列十一:事务管理(Spring事务管理的特点、事务概念学习、Spring事务使用学习、Spring事务管理API学习、Spring事务源码学习)

    一.Spring事务管理的特点 Spring框架为事务管理提供一套统一的抽象,带来的好处有:1. 跨不同事务API的统一的编程模型,无论你使用的是jdbc.jta.jpa.hibernate.2. 支 ...

随机推荐

  1. 常用类string的用法

    在Java中string是我们用的很多的一种类,下面就来说说string类中经常用到的一些方法. 1.string与数组相关的方法: 比如:string str = "fsafdsafdas ...

  2. Python Virtualenv运行Django环境配置

    系统: RHEL6.5 版本说明: Python-3.5.0 Django-1.10.4 virtualenv:为每个项目建立不同的/独立的Python环境,你将为每个项目安装所有需要的软件包到它们各 ...

  3. 模拟搭建Web项目的真实运行环境(四)

    本篇介绍如何部署mongodb环境,主要分为三个部分: 第一部分 介绍如何在ubuntu下安装mongodb, 第二部分 介绍如何在windows下安装使用MongoChef客户端, 第三部分 介绍在 ...

  4. Bill的挑战(bzoj 1879)

    Description Input 本题包含多组数据. 第一行:一个整数T,表示数据的个数. 对于每组数据: 第一行:两个整数,N和K(含义如题目表述). 接下来N行:每行一个字符串. Output ...

  5. 整数转IP地址

    将一个整数,比如1567898765转换为xxx.xxx.xxx.xxx的IP地址的形式, 以下是源代码 union IPNode{ unsigned int addr; struct { unsig ...

  6. LInux Shell 快捷键

    CTRL 键相关的快捷键: Ctrl + a - Jump to the start of the lineCtrl + b - Move back a charCtrl + c - Terminat ...

  7. 【Java EE 学习 74 上】【数据采集系统第六天】【使用Jfreechart的统计图实现】【Jfreechart的基本使用方法】

    之前已经实现了数据的采集,现在已经有了基本的数据,下一步就需要使用这些数据实现统计图的绘制了.这里使用Jfreechart实现这些统计图的绘制.首先看一下Jfreechart的基本用法,只有知道了它的 ...

  8. Redis到底该如何利用(三)?

    上两篇受益匪浅,秉着趁热打铁,不挖到最深不罢休的精神,我决定追加这篇.上一篇里最后我有提到实现分级缓存管理应该是个可行的方案,因此今天特别实践了一下.不过缓存分级之后也发现了一些问题,例如下图: 当a ...

  9. SQL谜题(加减符号替代)

    问题:将以下字符串”.1.2.3.4.5.6.7.8.9 = 1“中的符号点(.)更改为符号加(+)或符号(-),有多少种方法?请用SQL解决此问题 计算过程: CREATE TABLE #(VAL ...

  10. iOS 键盘遮挡输入 解决办法

    .初始化及添加通知观察者 - (void)viewDidLoad { [super viewDidLoad]; self.tableView = [[UITableView alloc] initWi ...