【JDBC系列】- 扩展提升学习

扩展提升学习

😄生命不息,写作不止
🔥 继续踏上学习之路,学之分享笔记
👊 总有一天我也能像各位大佬一样
🏆 博客首页   @怒放吧德德  To记录领地
🌝分享学习心得,欢迎指正,大家一起学习成长!

在这里插入图片描述

文章目录

  • 扩展提升学习
    • 简介
    • 1、获取自增长主键回显
      • (1)、操作过程
      • (2)、代码
      • (3)、代码解析
    • 2、批量插入数据优化
      • 优化前
      • 优化后
    • 3、事务
      • 事务的概念
      • 在JDBC中操作事务

简介

上一两篇文章介绍了jdbc的使用过程以及statement与prepareStatement的用法和优缺点,这次就接着来学习一下扩展提升。

1、获取自增长主键回显

在新增数据的时候,如果主键标识是自增,往往会返回本次插入的自增标识。然而JDBC是支持将自增主键返回回来。

(1)、操作过程

其使用流程与之前的并没有什么不同,都是从 注册驱动、建立连接、编写SQL语句、创建preparedStatement对象、发送SQL语句、解析结果集、释放资源 具体流程如下图所示。
在这里插入图片描述

(2)、代码

/**
 * @Author: lyd
 * @Description: 获取自增主键的id回显
 * @Date: 2023/8/12
 */
public class JDBCPSOtherPart {
    @SneakyThrows
    public static void main(String[] args) {
        // 注册驱动
        Class.forName("com.mysql.cj.jdbc.Driver");
        // 创建连接
        Connection connection = DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/cloud_user", "root", "12356");
        // 编写SQL语句
        String sql = "insert into tb_user(username,address) values(?,?);";
        // 创建preparedStatement
        // 这样并不会返回主键, 要想返回就需要携带另一个参数:Statement.RETURN_GENERATED_KEYS
        PreparedStatement preparedStatement = connection.prepareStatement(sql, Statement.RETURN_GENERATED_KEYS);
        // 占位符赋值
        preparedStatement.setObject(1, "lkj");
        preparedStatement.setObject(2, "beijing");
        // 发送sql并获取结果集
        int i = preparedStatement.executeUpdate(); // 返回的是几行受影响
        // 获取存储主键结果集对象/只有主键
        ResultSet generatedKeys = preparedStatement.getGeneratedKeys();
        generatedKeys.next();
        System.out.println("操作成功" + i + "行受影响, 主键id为: " + generatedKeys.getLong(1));
        // 关闭资源
        generatedKeys.close();
        preparedStatement.close();
        connection.close();
    }
}

(3)、代码解析

在上面的示例中,基本的方式是相同的,也是使用executeUpdate()来插入数据,在创建预编译对象的时候,需要在放入sql语句携带另一个参数:Statement.RETURN_GENERATED_KEYS,这个参数告诉JDBC在执行插入语句后返回生成的主键。之后,就可以使用 getGeneratedKeys() 方法获取包含生成的主键值的 ResultSet 对象(这个对象只有一行一列,只是存放主键)。我们先将游标往下移一个位置,然后可以使用 getLong(1) 来获取第一列的自增主键值。

我们可以看一下这个方法,这里官方是有标注这通过Statement.RETURN_GENERATED_KEYS or Statement.NO_GENERATED_KEYS 来判断是否需要返回自增的键值,其返回是一个ResultSet对象。
在这里插入图片描述

2、批量插入数据优化

在现实开发中,肯定会需要需要插入万条以上的数据,在本次学习中,我们采用jdbc进行批量插入。在不优化的情况下,运行起来的时间是需要很久的,性能也会很拉跨。

优化前

在没有优化的时候,我们就是在预编译Statement对象中放入不同的值,并且每次去执行一下sql,这样的执行方式会导致执行时间比较长,性能也比较差,因为每次的插入就需要一次IO,IO次数越多越会导致性能的底下。

核心的代码如下

for (int i = 0; i < 10000; i++) {
    preparedStatement.setObject(1, "batch" + i);
    preparedStatement.setObject(2, "hangzhou");
    // 发送sql并获取结果集
    preparedStatement.executeUpdate();
}

我们看一下运行结果,当运行的时候就需要好久才能够结束,我们可以看到,在执行1万次插入需要执行18.64s。
在这里插入图片描述

优化后

优化前的执行方式就相当于是for循环去执行了1万次插入,在sql语句中,其实批量插入是可以追加的形式,每次数据用","隔开。

INSERT INTO table (column1, column2)
VALUES
    ('value1_1', 'value2_1'),
    ('value1_2', 'value2_2'),
    ('value1_3', 'value2_3'),
    ...;

在使用jdbc也是可以的,JDBC提供了相关方法:addBatch()进行追加和executeBatch()去执行SQL来批量插入,这样的效果就比上面的好,优化前需要执行1万次IO,而采用批量插入就只需要执行1次IO。

for (int i = 0; i < 10000; i++) {
    preparedStatement.setObject(1, "opt" + i);
    preparedStatement.setObject(2, "hunan");
    // 发送sql并获取结果集
    preparedStatement.addBatch(); // 不直接执行插入,追加到后面
}
preparedStatement.executeBatch(); // 批量操作

我们执行一下并且查看一下执行速度。

在这里插入图片描述

然而我们会发现,奇怪了,我已经使用了批量插入,但是执行的时间还是那么久,这到底是为什么呢?别慌,这是因为,在使用批量插入的时候,他是需要一定条件的,首先是我们需要在路径后面追加一个**?rewriteBatchedStatements=true** 代表允许这样的操作。并且还需要注意,在语句后面不能加”;“,否则拼接出来的sql会是错误的。

在这里插入图片描述

我们可以看到,这回执行1万条数据的时间就大大缩短了。从刚开始的18秒多降到1秒都不到。然而在此笔者需要说一下,在日常开发的时候,能尽量减少IO次数的就尽量减少,比如批量删除,也是可以用这样的形式操作。当然减少IO次数的做法是需要因地制宜的。

3、事务

事务的目的是确保数据库在执行过程中始终保持一致性和完整性,即使发生了系统故障或其他问题。

事务的概念

在数据库中,事务通常具有以下四个特性,通常被称为 ACID 特性:

  • 原子性(Atomicity): 事务被视为不可分割的最小工作单位,要么全部成功完成,要么全部失败回滚。即,事务的操作要么全部执行,要么一个都不执行。

  • 一致性(Consistency): 事务在执行前后,数据库从一个一致性状态转换到另一个一致性状态。这意味着事务执行后,数据库的完整性约束仍然得到保持。

  • 隔离性(Isolation): 事务之间是相互隔离的,一个事务的操作不应该对其他事务产生影响。即,每个事务都应该像它是系统中唯一的事务一样。

  • 持久性(Durability): 一旦事务提交成功,其所做的更改将永久保存在数据库中,即使发生系统故障也不会丢失。

事务的使用场景包括数据库中的插入、更新、删除操作,以及需要确保数据一致性的任何操作。通过使用事务,可以确保数据库在任何情况下都能保持数据的正确性和完整性,从而提高数据的可靠性。

在JDBC中操作事务

在jdbc中,我们可以简单的使用事务,首先需要关闭自动提交,在执行SQL后,手动进行提交事务,如果抛出异常就在异常处理进行回滚。

我们可以看一下, 主要的使用方式

try{
	connection.setAutoCommit(false); // 关闭自动提交事务
	// 增删修改的SQL执行
	connection.commit(); // 提交事务
} catch (Exception e) {
	connection.rollback(); // 回滚事务
} finally {
	// 关闭资源
    connection.close();
}

代码如下

/**
 * @Author: lyd
 * @Description: JDBC事务的使用
 * @Date: 2023/8/12
 */
public class JDBCTransaction {
    public static void main(String[] args) {
        Connection connection = null;
        PreparedStatement preparedStatement = null;
        try {
            // 注册驱动
            Class.forName("com.mysql.cj.jdbc.Driver");
            // 创建连接
            connection = DriverManager.getConnection("jdbc:mysql://127.0.0.1:3306/cloud_user", "root", "12356");
            // 关闭自动连接
            connection.setAutoCommit(false);
            // 编写SQL语句
            String sql = "insert into tb_user(username,address) values(?,?);";
            // 创建preparedStatement
            // 这样并不会返回主键, 要想返回就需要携带另一个参数:Statement.RETURN_GENERATED_KEYS
            preparedStatement = connection.prepareStatement(sql, Statement.RETURN_GENERATED_KEYS);
            // 占位符赋值
            preparedStatement.setObject(1, "ooo");
            preparedStatement.setObject(2, "beijing");
            if (true) {
                throw new RuntimeException("异常");
            }
            // 发送sql并获取结果集
            int i = preparedStatement.executeUpdate(); // 返回的是几行受影响
            // 获取存储主键结果集对象/只有主键
            ResultSet generatedKeys = preparedStatement.getGeneratedKeys();
            generatedKeys.next();
            System.out.println("操作成功" + i + "行受影响, 主键id为: " + generatedKeys.getLong(1));
            connection.commit(); // 提交事务
        } catch (Exception e) {
            // 事务回滚
            System.out.println("事务回滚");
            try {
                connection.rollback(); // 回滚事务
            } catch (SQLException ex) {
                throw new RuntimeException(ex);
            }
        } finally {
            // 关闭资源
            try {
                preparedStatement.close();
                connection.close();
            } catch (SQLException e) {
                throw new RuntimeException(e);
            }
        }
    }
}

坚持学习,冲上顶峰!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/71753.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【前端】求职必备知识点4-CSS:flex、隐藏元素(7种方法)、单位

文章目录 flex隐藏元素&#xff08;7种方法&#xff09;不占位置占位置 单位思维导图 flex 【前端】CSS3弹性布局&#xff08;flex&#xff09;、媒体查询实现响应式布局和自适应布局_css媒体查询 自适应_karshey的博客-CSDN博客 flex缩写&#xff1a; flex-grow 和 flex-shr…

【Linux】网络基础2

文章目录 网络基础21. 应用层1.1 协议1.2 HTTP 协议1.2.1 URL1.2.2 urlencode和urldecode1.2.3 HTTP协议格式1.2.4 HTTP的方法1.2.5 HTTP的状态码1.2.6 HTTP 常见的header1.2.7 最简单的HTTP服务器 2. 传输层2.1 端口号2.1.1 端口号范围划分2.1.2 认识知名端口号2.1.3 netstat2…

浏览器无法连接网络问题

问题描述 电脑其他程序都能正常联网&#xff0c;但是所有的浏览器都无法联网&#xff0c;同时外部网站都能ping通 问题诊断 查看电脑Internet连接的问题报告显示&#xff1a;该设备或资源(Web 代理)未设置为接受端口"7890"上的连接。 解决方案 经过检查发现不是IP地址…

PHP Mysql查询全部全部返回字符串类型

设置pdo属性 $pdo->setAttribute(PDO::ATTR_EMULATE_PREPARES, true);

中科亿海微浮点数转换定点数

引言 浮点数转换定点数是一种常见的数值转换技术&#xff0c;用于将浮点数表示转换为定点数表示。浮点数表示采用指数和尾数的形式&#xff0c;可以表示较大范围的数值&#xff0c;但存在精度有限的问题。而定点数表示则采用固定小数点位置的形式&#xff0c;具有固定的精度和范…

架构训练营学习笔记:6-1 微服务

序 这部分是了解的。传统企业使用soa较多。很多企业银行、电信对于Oracle 依赖大&#xff0c;强调稳定性。各个项目侧重外包&#xff0c;技术栈不统一。 soa 历史 这个之前电信的BOSS系统就是这种架构&#xff0c;不知道现在呢&#xff0c;核心计费系统billing是运行在tuxduo…

1.作用域

1.1局部作用域 局部作用域分为函数作用域和块作用域。 1.函数作用域: 在函数内部声明的变量只能在函数内部被访问&#xff0c;外部无法直接访问。 总结&#xff1a; (1)函数内部声明的变量&#xff0c;在函数外部无法被访问 (2)函数的参数也是函数内部的局部变量 (3)不同函数…

分布式协调组件Zookeeper

Zookeeper介绍 什么是Zookeeper ZooKeeper 是⼀种分布式协调组件&#xff0c;用于管理大型主机。在分布式环境中协调和管理服务是一个复杂的过程。ZooKeeper 通过其简单的架构和 API 解决了这个问题。ZooKeeper 允许开发人员专注于核心应用程序逻辑&#xff0c;而不必担心应用…

大数据扫盲(1): 数据仓库与ETL的关系及ETL工具推荐

在数字化时代&#xff0c;数据成为了企业决策的关键支持。然而&#xff0c;随着数据不断增长&#xff0c;有效地管理和利用这些数据变得至关重要。数据仓库和ETL工具作为数据管理和分析的核心&#xff0c;将帮助企业从庞杂的数据中提取有价值信息。 一、ETL是什么&#xff1f; …

【脚踢数据结构】内核链表

(꒪ꇴ꒪ )&#xff0c;Hello我是祐言QAQ我的博客主页&#xff1a;C/C语言,Linux基础,ARM开发板&#xff0c;软件配置等领域博主&#x1f30d;快上&#x1f698;&#xff0c;一起学习&#xff0c;让我们成为一个强大的攻城狮&#xff01;送给自己和读者的一句鸡汤&#x1f914;&…

Remote Sensing,2023 | 基于SBL的分布式毫米波相干雷达成像的高效实现

Remote Sensing,2023 | 基于SBL的分布式毫米波相干雷达成像的高效实现 注1&#xff1a;本文系“无线感知论文速递”系列之一&#xff0c;致力于简洁清晰完整地介绍、解读无线感知领域最新的顶会/顶刊论文(包括但不限于 Nature/Science及其子刊; MobiCom, Sigcom, MobiSys, NSDI…

WEB集群——LVS-DR 群集、nginx负载均衡

1、基于 CentOS 7 构建 LVS-DR 群集。 2、配置nginx负载均衡。 一、 LVS-DR 群集 1、LVS-DR工作原理 LVS-DR&#xff08;Linux Virtual Server Director Server&#xff09; 名称缩写说明 虚拟IP地址(Virtual IP Address) VIPDirector用于向客户端计算机提供服务的IP地址真实…

TCP网络服务器设计

最近设计了一个网络服务器程序&#xff0c;对于4C8G的机器配置&#xff0c;TPS可以达到5W。业务处理逻辑是简单的字符串处理。服务器接收请求后对下游进行类似广播的发送。在此分享一下设计方式&#xff0c;如果有改进思路欢迎大家交流分享。 程序运行在CentOS7.9操作系统上&a…

【uniapp】uniapp设置安全区域:

文章目录 一、效果图:二、实现代码: 一、效果图: 二、实现代码: {"path": "pages/index/index","style": {"navigationStyle": "custom","navigationBarTextStyle": "white","navigationBarTitle…

Unity之ShaderGraph 节点介绍 UV节点

UV节点 Flipbook&#xff08;翻页或纹理帧动画&#xff09; Polar Coordinates&#xff08;将输入 UV 的值转换为极坐标。&#xff09; Radial Shear&#xff08;径向剪切变形&#xff09; Rotate&#xff08;将UV 的值旋转&#xff09; Spherize&#xff08;鱼眼镜头的球形变…

CentOS7连接网络

1.下载centos7镜像文件 2.安装centos7 3.修改网卡,ens33. 注意: 这里使用的是dhcp,设置IPADDR192.168.31.64一方面是为了后面使用crt或者MobaXterm连接,另一方面它和windows电脑的网卡要一致.这样才可以连接到网络.win r,输入cmd,打开命令窗口输入ipconfig.可以看到IPv4: 102…

windows安装apache-jmeter-5.6.2教程

目录 一、下载安装包&#xff08;推荐第二种&#xff09; 二、安装jmeter 三、启动jmeter 一、下载安装包&#xff08;推荐第二种&#xff09; 1.官网下载&#xff1a;Apache JMeter - Download Apache JMeter 2.百度云下载&#xff1a;链接&#xff1a;https://pan.baidu.…

FLStudio21水果最新中文版升级下载

FLStudio21最新中文版是一款非常专业的后期编曲音频处理软件&#xff0c;对于音乐编辑处理的领域内的人而言&#xff0c;是非常能够满足需求的一款工具。FL Studio21拥有强大且专业的创作工具&#xff0c;这是先进的创作工具&#xff0c;让你的音乐突破想象力的限制。FL Studio…

数据结构-栈的实现(C语言版)

前言 栈是一种特殊的线性表&#xff0c;只允许在固定的一端进行插入和删除的操作&#xff0c;进行数据插入和删除的一端叫做栈顶&#xff0c;另一端叫做栈底。 栈中的数据元素遵循后进先出的的原则。 目录 1.压栈和出栈 2. 栈的实现 3.测试代码 1.压栈和出栈 压栈&#xff…

C数据结构与算法——无向图(邻接矩阵) 应用

实验任务 (1) 掌握图的邻接矩阵存储及基本算法&#xff1b; (2) 掌握该存储方式下的DFS和BFS算法。 实验内容 实现图的邻接矩阵存储结构实现基于邻接矩阵的相关算法及遍历算法 实验源码 #include <malloc.h> #include <stdio.h>#define MAXSIZE 1000 #define …