隔离区别对待?如何捆绑?Java中的jdbc数据库事务及其隔离级别

简介: 隔离区别对待?如何捆绑?Java中的jdbc数据库事务及其隔离级别

文章目录


一、数据库事务简介

二、JDBC事务处理

三、事务的ACID属性

四、数据库的隔离级别

五、设置隔离级别


一、数据库事务简介


事务:一组逻辑操作单元,使数据从一种状态变换到另一种状态。


事务处理(事务操作):保证所有事务都作为一个工作单元来执行,即使出现了故障,都不能改变这种执行方式。当在一个事务中执行多个操作时,要么所有的事务都被提交(commit),那么这些修改就永久地保存下来;要么数据库管理系统将放弃所作的所有修改,整个事务回滚(rollback)到最初状态。


为确保数据库中数据的一致性,数据的操作应当是离散的逻辑单元:当它全部完成时,数据的一致性可以保持,而当这个单元中的一部分操作失败,整个事务应全部视为错误,所有从起始点以后的操作应全部回退到开始状态。


二、JDBC事务处理


数据一旦提交,就不可回滚。


数据什么时候意味着提交?

①当一个连接对象被创建时,默认情况下是自动提交事务:每次执行一个 SQL 语句时,如果执行成功,就会向数据库自动提交,而不能回滚。

②关闭数据库连接,数据就会自动的提交。如果多个操作,每个操作使用的是自己单独的连接,则无法保证事务。即同一个事务的多个操作必须在同一个连接下。


JDBC程序中为了让多个 SQL 语句作为一个事务执行:

①调用 Connection 对象的 setAutoCommit(false) 以取消自动提交事务

②在所有的 SQL 语句都成功执行后,调用 commit() 方法提交事务

③在出现异常时,调用 rollback() 方法回滚事务


若 Connection 没有被关闭,还可能被重复使用,则需要恢复其自动提交状态setAutoCommit(true)。尤其是在使用数据库连接池技术时,执行close()方法前,建议恢复自动提交状态。


案例:用户AA向用户BB转账100


测试类:


package jdbc;
import utils.jdbcUtils;
import java.sql.Connection;
import java.sql.SQLException;
/**
 * @Author: Yeman
 * @Date: 2022-01-30-16:04
 * @Description:
 */
public class UpdateDate2 {
    public static void main(String[] args) {
        Connection conn = null;
        try {
            //获取连接
            conn = jdbcUtils.getSqlConnection();
            //取消自动提交,开启事务
            conn.setAutoCommit(false);
            //预编译sql语句
            String sql1 = "update user_table set balance = balance - 100 where user = ?";
            jdbcUtils.updateDate(conn,sql1,"AA");
            //模拟网络异常
            System.out.println(10 / 0);
            String sql2 = "update user_table set balance = balance + 100 where user = ?";
            jdbcUtils.updateDate(conn,sql2,"BB");
            //若没有异常则提交
            conn.commit();
        } catch (Exception e) {
            e.printStackTrace();
            //若有异常则回滚
            try {
                conn.rollback();
            } catch (SQLException throwables) {
                throwables.printStackTrace();
            }
        }finally {
            //恢复自动提交
            try {
                conn.setAutoCommit(true);
            } catch (SQLException e) {
                e.printStackTrace();
            }
            //关闭conn连接
            jdbcUtils.closeResource(conn,null);
        }
    }
}


工具类:


package utils;
import java.io.InputStream;
import java.sql.*;
import java.util.Properties;
/**
 * @Author: Yeman
 * @Date: 2022-01-12-14:34
 * @Description:操作数据库的工具类
 */
public class jdbcUtils {
    /**
     * @Description 获取数据库连接
     * @Param []
     * @return java.sql.Connection
     **/
    public static Connection getSqlConnection() throws Exception{
        //1、加载配置文件
        InputStream is = ClassLoader.getSystemClassLoader().getResourceAsStream("jdbc.properties");
        Properties pros = new Properties();
        pros.load(is);
        //2、读取配置信息
        String user = pros.getProperty("user");
        String password = pros.getProperty("password");
        String url = pros.getProperty("url");
        String driverClass = pros.getProperty("driverClass");
        //3、加载驱动
        Class.forName(driverClass);
        //4、获取连接
        Connection conn = DriverManager.getConnection(url, user, password);
        return conn;
    }
    /**
     * @Description 关闭连接和Statement资源
     * @Param [conn, ps]
     * @return void
     **/
    public static void closeResource(Connection conn, Statement ps){
        try {
            if (ps != null) ps.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if (conn != null) conn.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    public static void closeResource(Connection conn, Statement ps, ResultSet res){
        try {
            if (ps != null) ps.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if (conn != null) conn.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if (res != null) res.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    /**
     * @Description 通用增删改操作2.0
     * @Param
     * @return
     **/
    public static int updateDate(Connection conn, String sql, Object... args) {
        PreparedStatement ps = null;
        try {
            //预编译sql语句,获取PreparedStatement实例
            ps = conn.prepareStatement(sql);
            //填充占位符
            for (int i = 0; i < args.length; i++) {
                ps.setObject(i + 1, args[i]);
            }
            //执行操作
            return ps.executeUpdate(); //返回操作的字段数,没有则为0
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            //关闭资源
            closeResource(null, ps);
        }
        return 0;
    }
}


三、事务的ACID属性


1、原子性(Atomicity) 原子性是指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。


2、一致性(Consistency) 事务必须使数据库从一个一致性状态变换到另外一个一致性状态。


3、隔离性(Isolation) 事务的隔离性是指一个事务的执行不能被其他事务干扰,即一个事务内部的操作及使用的数据对并发的其他事务是隔离的,并发执行的各个事务之间不能互相干扰。


4、持久性(Durability) 持久性是指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,接下来的其他操作和数据库故障不应该对其有任何影响。


四、数据库的隔离级别


对于同时运行的多个事务,当这些事务访问数据库中相同的数据时,如果没有采取必要的隔离机制,就会导致各种并发问题:


①脏读:对于两个事务 T1和T2,T1 读取了已经被 T2 更新但还没有被提交的字段。之后,若 T2 回滚,T1读取的内容就是临时且无效的。

②不可重复读:对于两个事务T1和T2,T1 读取了一个字段,然后 T2 更新了该字段。之后,T1再次读取同一个字段,值就不同了。

③幻读:对于两个事务T1和T2,T1 从一个表中读取了一个字段,然后 T2 在该表中插入了一些新的行。之后,如果 T1 再次读取同一个表,就会多出几行。


数据库事务的隔离性:数据库系统必须具有隔离并发运行各个事务的能力,使它们不会相互影响,避免各种并发问题。


一个事务与其他事务隔离的程度称为隔离级别。数据库规定了多种事务隔离级别,不同隔离级别对应不同的干扰程度, 隔离级别越高, 数据一致性就越好,但并发性越弱。


Mysql 4种隔离级别:


ab065698873c4a949fa9287c133b87c4.png


Mysql 支持 4 种事务隔离级别,默认的事务隔离级别为: REPEATABLE READ。


五、设置隔离级别


案例:一个线程查询,另一个线程更新,当更新的线程尚未提交时查询的为初始的值,提交后查询到的为更新后的值。(这里将隔离级别设置为READ_COMMITTED)


测试类:


package jdbc;
import javabean.User;
import utils.Query;
import utils.jdbcUtils;
import java.sql.Connection;
import java.util.List;
/**
 * @Author: Yeman
 * @Date: 2022-01-30-22:27
 * @Description:
 */
public class TransactionTest {
    public static void main(String[] args) {
        //用于更新的线程
        TransactionThread transactionThread = new TransactionThread();
        Thread thread = new Thread(transactionThread);
        thread.start();
        //用于查询的线程
        Connection conn = null;
        try {
            conn = jdbcUtils.getSqlConnection();
            String sql = "select user,password,balance from user_table where user = ?";
            //设置隔离级别为提交可读
            conn.setTransactionIsolation(Connection.TRANSACTION_READ_COMMITTED);
            List<User> query1 = Query.Query(conn, User.class, sql, "CC");
            query1.forEach(System.out :: println);
            //延迟5秒后再次查询
            Thread.sleep(5000);
            List<User> query2 = Query.Query(conn,User.class,sql,"CC");
            query2.forEach(System.out :: println);
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            jdbcUtils.closeResource(conn,null);
        }
    }
}


用于更新的线程类:


package jdbc;
import utils.jdbcUtils;
import java.sql.Connection;
/**
 * @Author: Yeman
 * @Date: 2022-01-30-22:47
 * @Description:
 */
public class TransactionThread implements Runnable{
    @Override
    public void run() {
        Connection conn = null;
        try {
            conn = jdbcUtils.getSqlConnection();
            //取消自动提交
            conn.setAutoCommit(false);
            String sql = "update user_table set balance = ? where user = ?";
            jdbcUtils.updateDate(conn,sql,5000,"CC");
            //设置延迟3秒后再提交
            Thread.sleep(3000);
            conn.commit();
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            jdbcUtils.closeResource(conn,null);
        }
    }
}


工具类utils同上案例


测试结果:


d3e21e1600ab4011b854048a97d6a728.png


相关文章
|
2月前
|
XML Java 数据库连接
性能提升秘籍:如何高效使用Java连接池管理数据库连接
在Java应用中,数据库连接管理至关重要。随着访问量增加,频繁创建和关闭连接会影响性能。为此,Java连接池技术应运而生,如HikariCP。本文通过代码示例介绍如何引入HikariCP依赖、配置连接池参数及使用连接池高效管理数据库连接,提升系统性能。
75 5
|
1天前
|
SQL Java 关系型数据库
使用 JDBC 实现 Java 数据库操作
JDBC(Java Database Connectivity)是 Java 提供的数据库访问技术,允许通过 SQL 语句与数据库交互。本文详细介绍了 JDBC 的使用方法,包括环境准备、编程步骤和完整示例。
22 7
|
1天前
|
SQL Java 数据库连接
【潜意识Java】MyBatis中的动态SQL灵活、高效的数据库查询以及深度总结
本文详细介绍了MyBatis中的动态SQL功能,涵盖其背景、应用场景及实现方式。
27 6
|
1天前
|
SQL Java 数据库连接
【潜意识Java】Java中JDBC过时方法的替代方案以及JDBC为什么过时详细分析
本文介绍了JDBC中一些常见过时方法及其替代方案。
19 5
|
1天前
|
前端开发 Java 数据库连接
Java后端开发-使用springboot进行Mybatis连接数据库步骤
本文介绍了使用Java和IDEA进行数据库操作的详细步骤,涵盖从数据库准备到测试类编写及运行的全过程。主要内容包括: 1. **数据库准备**:创建数据库和表。 2. **查询数据库**:验证数据库是否可用。 3. **IDEA代码配置**:构建实体类并配置数据库连接。 4. **测试类编写**:编写并运行测试类以确保一切正常。
10 2
|
29天前
|
Java 数据库连接 数据库
springboot java.lang.ClassNotFoundException: dm.jdbc.driver.DmDriver应该如何解决
通过上述步骤,可以有效解决Spring Boot项目中遇到的 `java.lang.ClassNotFoundException: dm.jdbc.driver.DmDriver`问题。确保在项目中正确添加达梦数据库的JDBC驱动依赖,并在配置文件中正确配置数据源信息,是解决此问题的关键。通过这些方法,可以确保Spring Boot项目能够正确连接达梦数据库并正常运行。
205 31
|
2月前
|
JSON Java 关系型数据库
Java更新数据库报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
在Java中,使用mybatis-plus更新实体类对象到mysql,其中一个字段对应数据库中json数据类型,更新时报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
232 4
Java更新数据库报错:Data truncation: Cannot create a JSON value from a string with CHARACTER SET 'binary'.
|
25天前
|
SQL 存储 Java
数据库———事务及bug的解决
事务的一些概念,并发事务以及并发事务引起的bug,脏读,不可重复读,幻读,数据库中的隔离级别,事务的简单应用
|
1天前
|
缓存 关系型数据库 MySQL
【深入了解MySQL】优化查询性能与数据库设计的深度总结
本文详细介绍了MySQL查询优化和数据库设计技巧,涵盖基础优化、高级技巧及性能监控。
10 0
|
28天前
|
存储 Oracle 关系型数据库
数据库传奇:MySQL创世之父的两千金My、Maria
《数据库传奇:MySQL创世之父的两千金My、Maria》介绍了MySQL的发展历程及其分支MariaDB。MySQL由Michael Widenius等人于1994年创建,现归Oracle所有,广泛应用于阿里巴巴、腾讯等企业。2009年,Widenius因担心Oracle收购影响MySQL的开源性,创建了MariaDB,提供额外功能和改进。维基百科、Google等已逐步替换为MariaDB,以确保更好的性能和社区支持。掌握MariaDB作为备用方案,对未来发展至关重要。
57 3