JDBC->SpringJDBC->Mybatis封装JDBC

一、JDBC介绍


Java数据库连接,(Java Database Connectivity,简称JDBC)是Java语言中用来规范客户端程序如何来访问数据库的应用程序接口,提供了诸如查询和更新数据库中数据的方法。JDBC也是Sun Microsystems的商标。我们通常说的JDBC是面向关系型数据库的。

二、主要用途


1.与数据库建立连接

2.发送 SQL 语句

3.处理结果

三、JDBC概述


1、什么是JDBC 

JDBC(Java Data Base Connectivity,  Java数据库连接) 是一种用于执行SQL语句的Java API,为多种关系数据库提供统一访问 ,它由一组用Java语言编写的类和接口组成。


2、有了JDBC,程序员只需用JDBC API写一个程序,就可访问所有数据库。(常用的框架Hibernate、MyBatis底层都是JDBC,都是对JDBC的封装) 


 3、 Sun公司、数据库厂商、程序员三方关系:SUN公司是规范制定者,制定了规范JDBC(连接数据库规范)

(1)DriverManager类   作用:管理各种不同的JDBC驱动

(2)Connection接口  

(3)Statement接口和PreparedStatement接口    

(4)ResultSet接口

数据库厂商微软、甲骨文等分别提供实现JDBC接口的驱动jar包,程序员学习JDBC规范来应用这些jar包里的类。


4、JDBC访问数据库步骤

  • :加载一个Driver驱动
  • :创建数据库连接(Connection)
  •  :创建SQL命令发送器Statement
  • :通过Statement发送SQL命令并得到结果
  • :处理结果(select语句)
  • :关闭数据库资源

四、实例:用JDBC完成查询操作

 

 

1、ResultSet里的数据一行一行排列,每行有多个字段,且有一个记录指针,指针所指的数据行叫做当前数据行,我们只能来操作当前的数据行。我们如果想要取得某一条记录,就要使用ResultSet的next()方法 ,如果我们想要得到ResultSet里的所有记录,就应该使用while循环。

2、ResultSet对象自动维护指向当前数据行的游标。每调用一次next()方法,游标向下移动一行。

3、初始状态下记录指针指向第一条记录的前面,通过next()方法指向第一条记录。循环完毕后指向最后一条记录的后面。

 ResultSet类的方法


4、作为一种好的编程风格,应在不需要Statement对象和Connection对象时显式地关闭它们。关闭Statement对象和Connection对象的语法形式为:

5、用户不必关闭ResultSet。当它的 Statement 关闭、重新执行或用于从多结果序列中获取下一个结果时,该ResultSet将被自动关闭

示例:是将后台查询到的数据在后台输出。实际开发中应该是客户端向服务器发起请求,服务器端访问数据库得到数据并封装到List中,作为整体传输到前台,并在前台输出。另外可以使用try-with-resources方式来处理异常而省略finally语句中对资源的手动关闭。 

五、JDBC高级操作


1、JDBC中使用事务

(1)在JDBC中,事务操作缺省是自动提交。一条对数据库的DML(insert、update、delete)代表一项事务操作,操作成功后,系统将自动调用commit()提交,否则自动调用rollback()回滚

(2)在JDBC中,事务操作方法都位于接口java.sql.Connection中,可以通过调用setAutoCommit(false)来禁止自动提交。之后就可以把多个数据库操作的表达式作为一个事务,在操作完成后调用commit()来进行整体提交,倘若其中一个表达式操作失败,都不会执行到commit(),并且将产生响应的异常;此时就可以在异常捕获时调用rollback()进行回滚,回复至数据初始状态

(3)事务开始的边界则不是那么明显了,它会开始于组成当前事务的所有statement中的第一个被执行的时候。

(4)事务结束的边界是commit或者rollback方法的调用

2、连接池

建立数据库连接的两种方式:

(1)传统连接方式:

首先调用Class.forName()方法加载数据库驱动,

然后调用DriverManager.getConnection()方法建立连接.

(2)连接池技术:

连接池解决方案是在应用程序启动时就预先建立多个数据库连接对象,然后将连接对象保存到连接池中。

当客户请求到来时,从池中取出一个连接对象为客户服务。

当请求完成时,客户程序调用close()方法,将连接对象放回池中.

对于多于连接池中连接数的请求,排队等待。

应用程序还可根据连接池中连接的使用率,动态增加或减少池中的连接数。 

 

3、传统的数据库连接方式的缺点

一个连接对象对应一个物理连接

每次操作都打开一个物理连接,

使用完都关闭连接,造成系统性能低下。

4、连接池技术的优点

客户程序得到的连接对象是连接池中物理连接的一个句柄

调用连接对象的close()方法,物理连接并没有关闭,数据源的实现只是删除了客户程序中的连接对象和池中的连接对象之间的联系.

5、数据库连接的建立及关闭是耗费系统资源的操作,在大型应用中对系统的性能影响尤为明显。为了能重复利用数据库连接对象,缩短请求的响应时间和提高服务器的性能,支持更多的客户,应采用连接池技术。

六、JDBC API总结


1、Connection接口

2、DriverManager类

作用:管理一组 JDBC 驱动程序的基本服务

应用程序不再需要使用 Class.forName() 显式地加载 JDBC 驱动程序。在调用 getConnection 方法时,DriverManager 会试着从初始化时加载的那些驱动程序以及使用与当前 applet 或应用程序相同的类加载器显式加载的那些驱动程序中查找合适的驱动程序。 

3、Statement接口

作用:用于将 SQL 语句发送到数据库中,或理解为执行sql语句

有三种 Statement对象:

Statement:用于执行不带参数的简单SQL语句;

PreparedStatement(从 Statement 继承):用于执行带或不带参数的预编译SQL语句;

CallableStatement(从PreparedStatement 继承):用于执行数据库存储过程的调用。

 

4、PreparedStatement接口

关系:public interface PreparedStatement extends Statement

区别

(1)PreparedStatment安全性高,可以避免SQL注入

(2)PreparedStatment简单不繁琐,不用进行字符串拼接

(3)PreparedStatment性能高,用在执行多个相同数据库DML操作时

5、ResultSet接口

ResultSet对象是executeQuery()方法的返回值,它被称为结果集,它代表符合SQL语句条件的所有行,并且它通过一套getXXX方法(这些get方法可以访问当前行中的不同列)提供了对这些行中数据的访问。

ResultSet里的数据一行一行排列,每行有多个字段,且有一个记录指针,指针所指的数据行叫做当前数据行,我们只能来操作当前的数据行。我们如果想要取得某一条记录,就要使用ResultSet的next()方法 ,如果我们想要得到ResultSet里的所有记录,就应该使用while循环。

ResultSet对象自动维护指向当前数据行的游标。每调用一次next()方法,游标向下移动一行。

初始状态下记录指针指向第一条记录的前面,通过next()方法指向第一条记录。循环完毕后指向最后一条记录的后面。

七、连接池体系

①传统连接方式的弊端分析

 1.局限性 : 

我们知道,JDBC连接MySQL本质上还是走了Socket连接,即网络通讯;当Java程序在同一时间间隔内要与数据库建立较多连接时,会消耗较长的时间,网络开销大。而且,Java程序通过JDBC连接数据库的最大连接数是有限制的,如果同一时间间隔内Java程序要建立很多与数据库的连接,甚至有多个Java程序都要与数据库建立连接,很可能会把数据库给干爆,导致数据库瘫痪,跑不起来;此时Java程序会抛出"Too many connections"异常。

2.几个弊端 :

  •  传统的JDBC连接方式是通过DriverManager类的getConnection方法来获取连接的;每次与数据库建立连接,都需要将获取到的Connection对象加载到内存中,再验证IP地址,用户名和密码等信息(通常消耗0.05~1s 时间,取决于此时计算机的网络卡顿情况和此时的并发线程数)。因此,频繁的建立与数据库的连接会占用很多的系统资源,容易造成服务器崩溃。
  • 每一次建立与数据库的连接后,使用完毕都必须断开连接;否则,如果程序出现异常而未能正常关闭,将导致数据库内存泄漏,最终将导致重启数据库。
  • 传统获取连接的方式,无法控制实际建立连接的数量,若连接过多,也可能导致内存泄漏,MySQL崩溃。

 

②数据库连接池

1.基本介绍 : 


数据库连接池是指,在Java程序未和数据库建立连接时,就提前创建好一定数量的连接并放入缓冲池中;当Java程序请求建立数据库连接时,就可以直接从缓冲池中“拿出”建立好的连接来用,用完后取消Java程序对该连接的引用即可,连接本身不会中断,只是“放回”了连接池(动态绑定机制)。数据库连接池负责分配,管理和释放数据库连接,它允许用户程序重复使用一个现有的数据库连接,而不是重新建立一个。(即连接池中的连接是公共的,谁都能用,你用完我可以接着用)当应用程序向连接池请求的连接数超过最大连接数量时,这些请求将被加入到等待队列中。

2.示意图如下 : 

3.连接池种类 : 

  • JDBC的数据库连接池使用javax.sql.DataSource来表示,DataSource只是一个接口,该接口通常由第三方来实现。
  • C3P0数据库连接池,速度相对较慢(只是慢一丢丢),但是稳定性很好,Hibernate,Spring底层用的就是C3P0。
  • DBCP数据库连接池,速度比C3P0快,但是稳定性差。
  • Proxool数据库连接池,有监控连接池状态的功能,但稳定性仍然比C3P0差一些。
  • BoneCP数据库连接池,速度较快。
  • Druid数据库连接池(德鲁伊连接池),由阿里提供,集DBCP,Proxool,C3P0连接池的优点于一身,是日常项目开发中使用频率最高的数据库连接池。

 八、SpringJDBC

1.Spring的JDBC模块有四个包组成:

  • core:核心包,JDBC核心功能,JdbcTemplate就在此包内
  • DataSource:数据源包,访问数据源的工具类,SimpleDriverDataSource 在此包
  • Object:对象包,以面向对象的方式访问数据库,它允许执行查询,并将结果作为业务对象,处理数据表和业务对象之间的映射
  • support:支持包,含有core和object包的支持类
     

2.配置数据库资源


在Spring中,通过JDBC驱动定义数据源是最简单的配置方式。Spring提供了三个这样的数据源类(均位于org.springframework.jdbc.datasource包中)供选择:

  • org.springframework.jdbc.datasource.DriverManagerDataSource:在每个连接请求时都会返回一个新建的连接,旧的JDBC,无池化管理。基本上已被SimpleDriverDataSource取代。
  • org.springframework.jdbc.datasource.SimpleDriverDataSource:与DriverManagerDataSource的工作方式类似,但是它直接使用JDBC驱动。
  • org.springframework.jdbc.datasource.SingleConnectionDataSource:它的连接不会关闭,在每个连接请求时都会返回同一个的连接。(它不是严格意义上的连接池数据源,但是可以将其视为只有一个连接的连接池)

除了这些简单的数据原外,所以我们也可以选择其他的第三方连接池使用,第三方的数据库连接池其实有很多,不过大多使用:Druid(阿里的)、HikariCP(springboot目前默认的)。
 

3. JdbcTemplate

在web编程中,传统JDBC每次执行SQL时每次都需要获取:Connection、PreparedStatement、ResultSet 这些数据库资源,然后就需要大量的try、catch、finally语句捕捉异常、关闭数据库资源。既便是专门使用一个类设置成单例进行封装,它依然是很繁琐的。在spring 中提供了JdbcTemplate模板来解决这个问题,它是比较经典的jdbc实现方式之一,同时也是Spring 在jdbc方便最底层的方法,SimpleJdbcInsert,SimpleJdbcCall 等底层都是用了JdbcTemplate。JdbcTemplate是 org.springframework.jdbc.core 中的类,用来处理数据库资源的创建和释放, JdbcTemplate类主要包含下面这些功能:

  • 执行SQL查询语句
  • 执行更新语句和存储过程调用
  • 对 ResultSet 执行迭代并提取返回的参数值。
  • 捕获JDBC异常并进行处理

我们发现这些功能都是我们在写 JCBC时每条sql语句都会涉及到的内容,而JdbcTemplate把这些提取出来进行封装,我们只需要在XML中配置好,就可直接提供sql语句执行了。

<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    <!--将之前配置好的连接池加载进来 -->
    <property name="dataSource" ref="druidDataSource"/>
</bean>

获取JdbcTemplate对象就可以直接使用,它内部的常用的方法

  • query():重载了大量 query方法来进行数据的查询操作,返回的是List,内部是自定义的Bean

List query(String sql, PreparedStatementSetter pss, RowMapper rowMapper):根据sql语句创建PreparedStatementSetter 对象。通过RowMapper 将结果返回到list中

List query(String sql, Object[] args, RowMapper rowMapper):使用Object[] 的值注入sql语句,利用RowMapper 返回数据

List query(String sql, RowMapper rowMapper, Object… args)

jdbcTemplate.query(sql, new BeanPropertyRowMapper<>(数据表对应的类.class),参数....);
  • queryForObject(String sql, RowMapper rowMapper, Object… args)():只能查询一行数据库记录,RowMapper自动封装,返回任意对象
jdbcTemplate.queryForObject(sql, new BeanPropertyRowMapper<>(数据表对应的类.class),参数....);
  • queryForList():可以返回多行数据的结果,返回的是List数据
  • update():此方法用来进行修改、新增、删除操作
int update(String sql)
int update(String sql,Object… args)
int[] batchUpdate(String sql, List<Object[]> batchArgs, final int[] argTypes) :批量执行新增、更新、删除等语句

sql:需要执行的 SQL 语句;argTypes:需要注入的 SQL 参数的 JDBC 类型;

batchArgs:表示需要传入到 SQL 语句的参数。

execute():可以执行任意 SQL,一般用于执行 DDL 语句;
@SpringJUnitConfig(locations = {"classpath:application.xml"})
public class JDBCTest {
   	@Resource
    private JdbcTemplate jdbcTemplate;
    
    @Test
    public void select(){
        Integer[] id = {1};
        String sql = "select * from classes where c_id =?";
        //Classes classes =jdbcTemplate1.query(sql, new BeanPropertyRowMapper<>(Classes.class),id); 
        Classes classes = jdbcTemplate.queryForObject(sql,id,(ResultSet rs,int rowNum)->{
            Classes classes1 = new Classes();
            classes1.setcId(rs.getInt("c_id"));
            classes1.setClassesName(rs.getString("classes_name"));
            classes1.setDeleted(rs.getBoolean("is_deleted"));
            return classes1;
        });
        if(classes!=null){
            System.out.println(classes.getClassesName());
        }
    }
     @Test
    public void selectAll(){
        String sql = "select * from classes";
        List<Classes> list = jdbcTemplate.query(sql, new BeanPropertyRowMapper<>(Classes.class));
        list.forEach(c -> System.out.println(c.getClassesName()));
    }

    @Test
    public void insert(){
        String classesName = "新增班级";
        String sql = "insert into classes(classes_name) values(?)";
        jdbcTemplate.update(sql, classesName);
        
    }
    @Test
    public void delete(){
        int id = 1;
        String sql = "delete from classes where c_id=?";
        jdbcTemplate.update(sql,id);
    }

    @Test
    public void update(){
        String newName = "修改班级";
        int id = 1;
        String sql = "UPDATE classes SET classes_name='?' WHERE classes.c_id = ?";
        jdbcTemplate.update(sql,newName,id);
    }
}

 九、spring整合mybatis

 

1.传统的dao方式操作数据库

即获取sqlSession,然后调用sqlSession.selectOne 等等的方式来进行数据库操作。

2.mapper方式

使用mapper方式,只需要一个接口类,不需要实现类,但有几点要求:

(1)映射文件保持命名空间与接口类名称一致,注意这个命名空间,如果没有在sqlMapConfig里配置好typeAliases,则必须是全路径命称,即此处命名空间名称必须为mapper接口路径加名称

(2)映射文件的SQLID名,与mapper接品方法名一致

(3)映射文件的输入参数与mapper接口的形参一致

(4)映射文件的返回结果类型与mapper接口的返回类型一致

另注意如果,在sqlMapConfig里采用的是包方式的映射(即mappers标签下,使用的是package标签),则需保证映射文件与mapper接口在同一目录下,并且必须名称相称,如果不是以包方式,则不需要如此,只需要保证以上4点即可。另第二种方式一般以sqlSession.getMapper 方式调用。

 

十、SpringBoot持久化

1.使用SpringBoot的过程中,常用的持久化解决方案主要有两种,一种是Mybatis框架,另一个就是SpringData JPA。

2. SpringData JPA和MyBatis最大的区别就是SpringData JPA是Spring亲生的,这个从名字的命名方式上也能看出来,当然是因为它们是一家人了。

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

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

相关文章

Leetcode每日一题周汇总 (12.24-12.30)

Leetcode每日一题周汇总 (12.24-12.30) 1.Sunday (12.24) 题目链接&#xff1a;1954. 收集足够苹果的最小花园周长 - 力扣&#xff08;LeetCode&#xff09; 题目描述&#xff1a; 给你一个用无限二维网格表示的花园&#xff0c;每一个 整数坐标处都有一棵苹果树。整数坐标…

今晚咱们一起来场特别的技术跨年!!

▼最近直播超级多&#xff0c;预约保你有收获 今晚跨年直播&#xff1a;《LLM在电商推荐系统的应用案例实战》 —1— 今晚咱们来场技术跨年&#xff01; LLM 大模型无疑是2023年最重磅的技术&#xff0c;逐渐在各行各业产生了越来越重要的实质影响&#xff0c;2024年的钟声今晚…

vue3-12

需求是用户如果登录了&#xff0c;可以访问主页&#xff0c;如果没有登录&#xff0c;则不能访问主页&#xff0c;随后跳转到登录界面&#xff0c;让用户登录 实现思路&#xff0c;在用户登录之前做一个检查&#xff0c;如果登录了&#xff0c;则token是存在的&#xff0c;则放…

Android Studio如何查找和替换

目录 前言 一、概述 二、总结 三、更多资源 前言 在Android Studio中&#xff0c;查找和替换是非常常见的操作&#xff0c;它可以帮助我们快速地定位和修改代码中的错误或不合适的内容。本文将介绍如何在Android Studio中进行查找和替换操作&#xff0c;包括基本的查找和替…

蓝牙音频发射模块 蓝牙耳机连接是如何操作的以BT321F为例

一、简介 蓝牙发射模块 蓝牙耳机连接是如何操作的以BT321F模块为例 因为蓝牙音频类的产品&#xff0c;基本上大体分为三类 第一类&#xff1a;蓝牙耳机&#xff0c;包括&#xff1a;tws无线耳机&#xff0c;如airpods&#xff0c;头戴耳机如bose qc35系列等等 第二类&#…

QtitanRibbon 开始使用实例

新建一个界面程序&#xff1a; 修改项目里面的源码&#xff1a; 至此&#xff0c;一个简单界面就出来了&#xff0c;效果如下所示&#xff1a;

STM32CubeMX教程15 ADC - 多重ADC转换

目录 1、准备材料 2、实验目标 3、二重ADC转换 3.0、前提知识 3.1、CubeMX相关配置 3.1.1、时钟树配置 3.1.2、外设参数配置 3.1.3、外设中断配置 3.2、生成代码 3.2.1、外设初始化调用流程 3.2.2、外设中断调用流程 3.2.3、添加其他必要代码 4、常用函数 5、烧录…

基于 LightGBM 的系统访问风险识别

基于 LightGBM 的系统访问风险识别 文章目录 基于 LightGBM 的系统访问风险识别一、课题来源二、任务描述三、课题背景四、数据获取分析及说明&#xff08;1&#xff09;登录https://www.datafountain.cn并获取相关数据&#xff08;2&#xff09;数据集文件说明&#xff08;3&a…

UDS诊断(ISO14229-1) 11服务

文章目录 功能简介应用场景请求和响应1、请求2、子功能3、肯定响应4、否定响应 报文示例UDS中常用 NRC参考 功能简介 11服务&#xff0c;即 ECUReset&#xff08;ECU重置&#xff09;服务&#xff0c;用来控制MCU进行重启&#xff0c;重启分为硬件重启和软件重启。 应用场景…

TikTok真题第11天 | 1249.移除无效的括号、23.合并K个升序链表、773.滑动谜题

今天开始整hard题&#xff0c;果然费时。 1249.移除无效的括号 题目链接&#xff1a;1249.minimum-remove-to-make-valid-parentheses 解法&#xff1a; 这个题用栈来处理&#xff0c;用栈来记录左括号的位置&#xff0c;同时用一个向量来记录左括号和右括号是否有效&#x…

【Java系列】Iterator

Iterator&#xff08;迭代器&#xff09; Java Iterator&#xff08;迭代器&#xff09;迭代器接口定义了几个方法&#xff0c;最常用的是以下三个&#xff1a; Iterator 类位于 java.util 包中&#xff0c;使用前需要引入它&#xff0c;语法格式如下&#xff1a;获取一个迭代器…

第14课 多维数组

文章目录 前言一、多维数组的定义二、多维数组的初始化三、多维数组的使用&#xff08;以二维数组为例&#xff09;1. 矩阵转置问题 三、课后练习1. 求一个m*n矩阵中所有元素的累加和2. 查找并输出一个m*n矩阵中的最小元素以及其在矩阵中的位置3. 将m*n矩阵A复制为m*n矩阵B&…

2024年软件测试面试笔记(超详细整理)

大家好&#xff0c;最近有不少小伙伴在后台留言&#xff0c;得准备年后面试了&#xff0c;又不知道从何下手&#xff01;为了帮大家节约时间&#xff0c;特意准备了一份面试相关的资料&#xff0c;内容非常的全面&#xff0c;真的可以好好补一补&#xff0c;希望大家在都能拿到…

在宝塔Linux中安装Docker

前言 帮助使用宝塔的用户快速上手docke的安装 &#x1f4da;&#x1f4da; &#x1f3c5;我是默&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; ​​​​ &#x1f31f;在这里&#xff0c;我要推荐给大家我的专栏《Docker》。&#x1f3af;&#x1f3af…

大数据 - Hadoop系列《三》- HDFS(分布式文件系统)概述

&#x1f436;5.1 hdfs的概念 HDFS分布式文件系统,全称为:Hadoop Distributed File System。 它是一个文件系统&#xff0c;用于存储文件&#xff0c;通过目录树来定位文件&#xff1b;其次&#xff0c;它是分布式的&#xff0c;由很多服务器联合起来实现其功能&#xff0c;集…

2024年【高压电工】找解析及高压电工考试技巧

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 高压电工找解析根据新高压电工考试大纲要求&#xff0c;安全生产模拟考试一点通将高压电工模拟考试试题进行汇编&#xff0c;组成一套高压电工全真模拟考试试题&#xff0c;学员可通过高压电工考试技巧全真模拟&#…

Android 反编译处理Dex

前言 当我们将Android项目打包上架的时候&#xff0c;为了提高被人反编译代码的可能性可以提取 dex 文件对代码进一步做混淆处理。 本文不对相关工具做过多的解释&#xff0c;不了解的可以先熟悉相关工具的使用。 相关工具&#xff08;点击直接下载&#xff09; jadx-gui&a…

IIS服务器发布PHP网站

IIS服务器&#xff0c;相信开发者都不会陌生&#xff0c;它的英文全称是Internet Information Services&#xff0c;是由微软公司提供的基于运行Microsoft Windows的互联网基本服务&#xff0c;常用于Windows系统的Web项目部署&#xff0c;本篇以PHP项目为例&#xff0c;讲解如…

使用 extract + TextMapAdapter 实现了自定义 traceId

前言 某些特定的场景&#xff0c;需要我们通过代码的方式实现自定义 traceId。 实现思路&#xff1a;通过 tracer.extract 能够构造出 SpanContext &#xff0c;将构造出来的 SpanContext 作为上层节点信息&#xff0c;通过 asChildOf(SpanContext) 能够构造出当前的 span。 …

SpringBoot + Vue 抖音全平台项目

简介 本项目是一个短视频平台&#xff0c;拥有热度排行榜&#xff0c;热门视频&#xff0c;兴趣推送&#xff0c;关注推送&#xff0c;内容审核等功能。 源码下载 网盘 (访问密码: 8418) 登录/注册 首页 创作中心 架构设计 上传视频业务流程 视频推送流程 1.用户订阅分类后…
最新文章