Java学习笔记21——使用JDBC访问MySQL数据库

JDBC(Java Database Connectivity,Java数据库连接)是应用程序编程借口(API),描述了一套访问关系数据库的标准Java类库。可以在程序中使用这些API,连接到关系数据库,执行SQL语句,对数据进行处理。JDBC不但提供了访问关系数据库的标准API,还为数据库厂商提供了一个标准的体系结构,让厂商可以为自己的数据库提供JDBC驱动程序,这些驱动程序可以让Java应用程序直接访问厂商的数据库产品,从而提高了Java程序访问数据库的效率。

1、JDBC驱动程序的类型:

JDBC驱动程序可以分为4类,分别是:

  • 1)JDBC-ODBC桥:通过ODBC访问数据库。
  • 2)部分本地API,部分Java驱动程序:大都C语言编写,依赖具体的平台。
  • 3)JDBC网络纯Java驱动程序:利用中间件的应用服务器来访问数据库,应用服务器作为数据库的网关,客户端通过它可以连接到不同的数据库服务器,如WebLogic服务器。
  • 4)本地协议纯Java驱动程序:目前,绝大多数数据库厂商支持客户程序通过网络直接与数据库通信的网络协议来直接访问数据库。这种类型的JDBC驱动程序完全用Java编写,通过与数据库建立直接的套接字连接,采用具体厂商的网络协议把JDBC API调用转换为直接的网络调用。这种类型的驱动是访问数据库效率最高的,不过,由于每个数据库厂商有自己的协议,因此访问不同厂商的数据库,需要不同的JDBC驱动程序。

2、下载MySQL JDBC 驱动

MySQL的JDBC 驱动程序没有包含在数据库的安装包中,需要单独下载,下载的网址为:MySQL :: Download Connector/Jicon-default.png?t=N7T8https://dev.mysql.com/downloads/connector/j/在浏览器中输入上面的网址,进入MySQL的JDBC 驱动程序的下载页面,如下图:

在上图下拉框中选择“Platfrom Independent”,然后出现如下图,Window平台下载zip文件下载。

将zip文件解压缩,将出现一个类似mysql-connector-j-8.3.0.jar的文件,它就是MySQL数据库的JDBC驱动。

3、JDBC API

JDBC API包含在JDK中,被分为两个包:java.sql和javax.sql。java.sql包中定义了访问数据库的接口和类,其中一些接口由驱动程序提供商来实现。

在开发数据库访问程序时,经常会编写如下的代码。

String url = "jdbc:mysql://127.0.0.1:3306/mysql";     
String username = "root"; 
String password = "2005Liuzx0302";// 加载数据库驱动
Class.forName("com.mysql.jdbc.Driver");		
conn = DriverManager.getConnection(url,username,password);
Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("select * from user_info");

从上面代码可以看出利用JDBC访问数据库有统一的步骤。

1)首先调用Class类的forName()方法加载要访问的数据库的JDBC驱动类,

2)接下来调用Java的JDBC API(java.sql包)中的DriverManager类中的getConnection方法建立与数据库的连接,

3)然后调用Connection对象的createStatement方法,

4)最后调用Statement对象的executeQuery方法执行SQL语句得到ResultSet对象,就可以利用该对象来获取数据库中的数据了。

当然,最后要关闭数据库连接对象,释放数据库连接资源。

4、加载并注册数据库驱动

使用JDBC方法数据库,首先就是要调用Class类的forName方法加载要访问数据库厂商的JDBC驱动。了解这行代码背后发生的事情让我感觉有些豁然开朗。

1)Driver接口

java.sql.Driver是所有JDBC驱动程序需要实现的接口。这个接口是提供给数据库厂商使用的,不同厂商实现该接口的类名是不同的。常见的有:

  • com.microsoft.jdbc.sqlserver.SQLServerDriver:微软SQL Server2000版本驱动
  • com.microsoft.sqlserver.jdbc.SQLServerDriver:微软SQL Server2005及以后版本驱动
  • oracle.jdbc.driver.OracleDriver:是Oracle的JDBC驱动的类名,Oracle安装目录下jdbc目录中
  • com.mysql.cj.jdbc.Driver:是MySQL 8.0.x版本的驱动的类名。

Driver接口提供了一个connection方法,用来建立到数据库的连接,该方法的签名如下:

Connection connect(String url,Properties info) throws SQLException

开发人员在程序中不需要直接去访问这些实现了Driver接口的类,而是由Java的JDK 中的JDBC API(java.sql)中的驱动程序管理器类提供的方法来建立数据库连接,通过驱动程序管理器类的getConnection()方法来获取数据库连接。

2)深入理解JDBC的加载与注册驱动背后发生的事情

加载JDBC驱动是调用Class类的静态方法forName,向其传递要加载的JDBC驱动的类名。在运行时,类加载从CLASSPATH路径中定位和加载JDBC驱动类。在加载驱动程序后,需要注册驱动程序的一个实例。

DriverManager类是属于JDBC API(java.sql包)中提供的,用来负责管理加载的驱动程序,这个类中的所有方法都是静态的,在DriverManager类中提供了registerDriver方法来注册驱动程序类的实例,该方法的签名如下:

public statcic void registerDriver(Driver driver) throw SQLException

需要清楚的是,通常不需要开发人员在程序中直接调用registerDriver方法来注册驱动程序的实例,因为实现Driver接口的驱动程序类都包含了一个静态代码块,在这个静态代码块中,会调用DriverManager.registerDriver方法来注册自身的的一个实例。当使用Class类的forName方法在数据库驱动程序类的时候,会自动执行该静态代码块,从而注册驱动程序的一个实例,这样使用DriverManager.getConnection方法,就返回相应的驱动程序的数据库连接(Connection)对象。

下面是MySQL 驱动程序类的代码片段:

public class Driver extends NonRegisteringDriver implements java.sql.Driver {//// Register ourselves with the DriverManager//static {try {java.sql.DriverManager.registerDriver(new Driver());} catch (SQLException e) {throw new RuntimeException("Can't register driver!");}}...}

从上面代码看,加载数据库厂商的驱动类,会自动执行该类的静态代码块,调用JDBC的驱动管理类DriverManager.registerDriver方法,并传入参数(new Driver())注册该厂商自身的驱动实例。

5、建立数据库连接

调用JDBC的DriverManager类的getConnection方法建立数据库连接,上面提到DriverManager类在java.sql包中,其中方法都是静态的,所以可以直接通过类名DriverManager.getConnection()来调用。在DriverManager类中重载了3个getConnection方法:

1)public static Connection getConnection(String url) throws SQLException

该方法通过给出的数据库URL建立到数据库的连接。

2)public static Connection getConnection(String url,String user,String password) throws SQLException

该方法除了需要数据库URL,还需要用户名和密码来建立到数据库的连接。

3)public static Connection getConnection(String url,Properties info) throws SQLException

该方法需要数据库URL和java.util.Properties对象。Properties对象包含了用于特定数据库所需要的参数,以键值对的方式指定连接参数,来建立到数据库的连接。

JDBC URL用于标识一个被注册的驱动程序,驱动程序管理器通过这个URL来选择正确的驱动程序,从而建立到数据库的连接。JDBC URL的语法如下:

jdbc:subprotocol:subname

整个URL用冒号(:)被分成了3个部分。

协议:在上面语法中,jdbc为协议,在JDBC中,它是唯一允许的协议,即总是它,只能是它。

子协议(subprotocl):子协议用于标识一个数据库驱动程序。

子名称(subname):子名称的语法与具体的驱动程序相关,不同驱动语法不尽相同。

常用数据可的JDBC URL:

SQL Server2005及以后:jdbc:sqlserver://localhost:1433;databasename=pubs

Oracle9i、10g、11g:jdbc:oracel:thin@localhost:1521:ORCL

MySQL8.0.x:jdbc:mysql://localhost:3306/databasename?userSSL=false&serverTimezone=UTC

  • userSSL=false: 这个参数指示JDBC驱动程序不要使用SSL来加密与MySQL服务器的连接。在某些情况下,为了简化配置或出于性能考虑,你可能想要禁用SSL。但请注意,禁用SSL可能会降低连接的安全性。
  • serverTimezone=UTC: 这个参数设置MySQL服务器使用的时区为UTC(协调世界时)。这有助于避免在Java应用程序和MySQL服务器之间由于时区差异而可能产生的时间问题。

serverTimezone=UTC 并不是JDBC连接MySQL时必须的参数。这个参数的主要用途是告诉JDBC驱动程序期望从MySQL服务器接收的时间戳使用哪个时区。如果不指定这个参数,JDBC驱动程序通常会使用系统的默认时区来解析时间戳,这可能导致在Java应用程序和MySQL服务器之间出现时区相关的问题。

如果你确定你的Java应用程序和MySQL服务器使用的时区是一致的,或者你愿意让Java应用程序使用系统的默认时区来解析时间戳,那么你可以省略这个参数。

然而,在实际应用中,为了避免潜在的时区问题,很多开发者会选择明确指定 serverTimezone 参数。特别是当你的应用程序需要处理跨时区的数据,或者当你的应用程序和数据库服务器部署在不同的地理位置时,指定时区参数会是一个好习惯。

另外,需要注意的是,除了 serverTimezone 参数,MySQL JDBC驱动程序还提供了其他与时区相关的参数,如 useLegacyDatetimeCode=false 和 useSSL=false 等。这些参数可以根据你的具体需求进行配置。

总的来说,是否指定 serverTimezone=UTC 取决于你的具体应用场景和需求。如果你不确定是否需要它,或者你的应用程序可能会受到时区问题的影响,那么建议添加这个参数以确保时间戳的正确处理。

6、访问数据库

数据库连接被用于向数据库服务器发送命令和SQL语句,在连接建立后,需要对数据库进行访问,执行SQL语句。在java.sql包中提供了3个接口,分别定义了对数据库调用的不同方式,分别是:Statement、PreparedStatement、CallableStatement。

1)Statement

调用Connnect对象的createStatement方法创建一个Statement对象。Statement对象用于执行静态的SQL语句,返回执行的结果。在Connection接口中定义了createStatement()方法,如下:

Statement createStatement() throws SQLException,常用来执行没有参数的SQL语句。

Statement接口中定义了如下的方法执行SQL语句:

  • ResultSet executeQuery(String sql) throws SQLException,该方法执行参数sql指定的SQL语句,返回一个ResultSet对象。ResultSet对象用于查看执行的结果。
  • int exceuteUpdate(String sql) throws SQLException,该方法执行参数sql指定的insert 、update或delete语句。该方法也可以用来执行SQL DDL 语句,必须创建表。
  • boolean execute(String sql) throws SQLException,该方法执行返回多个结果集的SQL语句。在某些情况下,一条SQL语句可以返回多个结果集或者更新行数。在通常情况下,不使用这个方法。除非要执行返回多个结果集的存储过程或者动态执行一个未知的SQL串。如果使用该方法,就必须使用getResultSet()或者getUpdateCount()方法获取结果,并且调用getMoreResults()方法来访问下一个结果集。如果返回的第一个结果是一个ResultSet对象,那么这个方法返回的是true;如果返回的是一个更新的行数或者没有结果,那么返回的是false
  • int[] executeBatch() throws SQLException,该方法允许向数据库提交一批命令,然后一起执行。如果所有的命令都成功执行,那么返回值是一个更新行数的数组。数组中的每一个int元素都是按照加入命令的先后顺序来存储的,表示了相应命令的更新行数。可以使用addBatch()方法将SQL命令加入到命令列表中。

2)PreparedStatement

在程序中传递的SQL语句在执行前需要被预先编译,包括语句分析、代码优化等,然后才能被数据库引擎执行。如果重复执行只有参数不同的SQL语句,则程序效率会比较低。这就是PreparedStatement使用比Statement多的原因。

如果要用不同的参数来多次执行同一个SQL语句,推荐使用PreparedStatement对象。PreparedStatement接口是继承Statement接口,它的对象表示一条预编译过的SQL语句。PreparedStatement对象所代表的SQL语句中的参数使用问号(?)来表示,调用PreparedStatement对象的setXxx方法来设置这些参数。setXxx方法有两个参数,第一个参数是要设置SQL语句中的参数索引(从1开始),第二个参数是要设置的SQL语句中的参数的值。

conn = DriverManager.getConnection(url,user,password);
Statement stmt = conn.createStatement();
stmt.executeUpdate("create table emp(id,INT,name VARCHAR(10),hiredate DATE");PreparedStatement pstmt = conn.prepareStatement("insert emp(?,?,?)");pstmt.setInt(1,1);
pstmt.setString(2,"liuzx");
pstmt.setDate(3,java.sql.Date.valueOf("2005-3-3"));
pstmt.executeUpdate();pstmt.setInt(1,2);
pstmt.setString(2,"renrui");
pstmt.setDate(3,java.sql.Date.valueOf("1976-9-19"));
pstmt.executeUpdate();

针对不同类型的参数,需要使用对应的setXxx方法,尤其对应数据库日期类型的Java类型是java.sql.Date而不是java.util.Date(虽然java.sql.Date是java.util.Date的子类),MySQL数据库中的数据类型与Java中的数据类型有一定的对应关系,需要注意。

3)ResultSet

ResultSet对象以逻辑表格的形式封装了执行数据库操作的结果集,ResultSet对象维护了一个指向当前数据行的游标,在初始的时候,游标在第一行之前,调用ResultSet对象的next方法移动到游标的第一行,next方法移动到游标的下一行,如果新的数据行有效(有数据),则返回true,否则返回false。通过判断这个方法的返回值,可以循环读取结果集中的数据库,直到最后。

该方法如下:

boolean next() throws SQLException

在ResultSet接口中定义了很多方法来获取当前行中列的数据,根据表中字段类型的不同,采用不同的方法来获取数据。如:getBoolean()、getInt()、getLong()、getString()、getDate()等。

在上面这些方法中,提供了两种形式的调用:一种是以列的索引作为参数(索引值从1开始),另一种是以列的名字作为参数。例如对于getString()方法,有下面两种形式:

  • String getString(int columnIndex) throws SQLException
  • String getString(String columnName) throws SQLException

如果不知道要获取的列数据的类型,则可以一律采用getString()方法来得到一个String类型的数据。

3.1)对于要遍历结果集ResultSet,利用ResultSet对象的next方法,循环取出每行的数据。需要注意的是,在ResultSet对象初始化的时候,ResultSet对象的游标指向第一行之前,所以需要先调用一次next方法(通常在while循环的条件判断中调用),将游标移动到第一行,读取最后一行后,再次执行next方法则返回false,while循环也就结束,完成了ResultSet结果集的遍历(从第一行到最后一行)。

try(rs) {while(rs.next()) {System.out.print(rs.getString("title") + "\t");System.out.print(rs.getString("author") + "\t");System.out.print(rs.getFloat("price") + "\t");System.out.println();}
}

在Java中,当你看到try后面直接跟着一对圆括号并包含一个或多个对象时,这通常是Java 7引入的“try-with-resources”语句的用法。这个特性主要用于自动管理实现了AutoCloseableCloseable接口的资源,如数据库连接、文件流等,确保在try块执行完毕后,这些资源能够被正确地关闭,即使发生异常也是如此。

AutoCloseable接口是java.lang包中的一个标记接口,它定义了一个close()方法,用于释放资源。任何实现了这个接口的对象都可以用在try-with-resources语句中。

在上面的代码中,当try块执行完毕后,rs对象的close()方法会自动被调用,无论是否发生了异常。这使得资源的管理更加简洁和安全,避免了因为忘记关闭资源而导致的潜在问题,如内存泄漏或文件句柄泄漏等。

这种结构特别适用于那些需要在finally块中关闭的资源,因为它自动处理了资源的关闭逻辑,使得代码更加清晰和易于维护。

3.3)多个条件的SQL语句,在应用中动态构造查询语句,往往可能需要多个不同条件,因此需要进行判断,并在程序中拼接and条件,可是while子句后的第一个条件是没有and关键字的,所以往往做法是判断是否有一个条件了,后面才好拼接“and”字符串。为了避免复杂的if判断,通常做法是在where子句后附加一个恒等条件:1=1,这样就简化了条件的字符串拼接,只要某个条件存在,就统一拼接:and  field=value,例如:

StringBuffer sb = new StringBuffer("select * from bookinfo where 1=1");
// and前有一个空格
sb.append(" and title=" + title);
sb.append(" and author=" + authoor);ResultSet rs = stmt.executeQuery(sb.toString());

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【技术栈】Redis 中的事务及持久化方式

SueWakeup 个人主页: SueWakeup 系列专栏:学习技术栈 个性签名:保留赤子之心也许是种幸运吧 本文封面由 凯楠📸 友情提供 目录 相关传送门 1. Redis 中的事务 2. Redis 持久化 2.1 RDB 方式 2.1.1 RDB手动 2.1.2 RDB自动 2.…

跨越时空的纽带:探索Facebook如何连接人与人

引言 Facebook作为全球最大的社交媒体平台之一,已经成为了人们日常生活中不可或缺的一部分。它不仅仅是一个社交网络,更是连接人与人、人与世界的纽带。在这篇文章中,我们将深入探讨Facebook如何跨越时空,连接人与人之间的关系&a…

Flutter 初始WidgetState 简单应用案例分析

本系列文章主要整理Flutter的知识汇总,由浅入深,从Widget的搭建到其中的原理。本文还是围绕Widget在开发中应用和理解。 关于Flutter环境配置和首次创建可以参考前面文章。链接如下: Flutter 安装部署与认识Dart语言 Flutter 使用AndroidS…

Spring Cloud 整合 GateWay

目录 第一章 微服务架构图 第二章 Spring Cloud整合Nacos集群 第三章 Spring Cloud GateWay 第四章 Spring Cloud Alibaba 整合Sentinel 第五章 Spring Cloud Alibaba 整合SkyWalking链路跟踪 第六章 Spring Cloud Alibaba 整合Seata分布式事务 第七章 Spring Cloud 集成Auth用…

多模匹配算法AC算法和单模匹配算法BM

多模匹配算法之AC算法详解 算法概述  Aho-Corasick算法 - 这是一种字典匹配算法,它用于在输入文本中查找字典中的字符串。时间复杂度是线性的。该算法应用有限自动机巧妙地将字符比较转化为了状态转移。  该算法的基本思想 − 在预处理阶段,AC自动机算法建立…

Springboot通过注解+切面实现接口权限校验

Springboot通过注解+切面实现接口权限校验 主要说一下在对接口请求时,如何用注解切面去拦截校验当前登录用户是否有访问权限 1.首先创建注解 HasPermission ,跟普通注解创建方式基本一致 Retention(RetentionPolicy.RUNTIME) Target(Element…

小火星露谷管理器 报错:“你似乎没有安装Edge的webview2”

错误 解决办法 你可以到这个地方下载安装webview2 https://developer.microsoft.com/zh-cn/microsoft-edge/webview2/?formMT00IS

2024年亚洲图像处理趋势会议(ATIP 2024)即将召开!

2024年亚洲图像处理趋势会议(简称:ATIP 2024)将于2024年6月21日至23日在英国伦敦举行。在会议上我们将与相关领域的研究人员和知名专业人士共同讨论关于图像处理学科的最新研究方向及进展,评估当前最先进的技术和未来研究的关键领…

Tomcat(Win+Linux)安装教程

Windows环境安装 Tomcat安装及配置教程主要分为四步: 步骤一:确认自己是否已 安装JDK🔍 步骤二:下载安装Tomcat 步骤三:Tomcat配置环境变量 步骤四:验证Tomcat配置是否成功 OK,我们开始&#x…

数据库基本介绍及编译安装mysql

目录 数据库介绍 数据库类型 数据库管理系统(DBMS) 数据库系统 DBMS的工作模式 关系型数据库的优缺点 编译安装mysql 数据库介绍 数据:描述事物的的符号纪录称为数据(Data) 表:以行和列的形式组成…

python大学生健身爱好者交流网站flask-django-nodejs-php

任何系统都要遵循系统设计的基本流程,本系统也不例外,同样需要经过市场调研,需求分析,概要设计,详细设计,编码,测试这些步骤,基于python技术、django/flask框架、B/S机构、Mysql数据…

【No.13】蓝桥杯二分查找|整数二分|实数二分|跳石头|M次方根|分巧克力(C++)

二分查找算法 知识点 二分查找原理讲解在单调递增序列 a 中查找 x 或 x 的后继在单调递增序列 a 中查找 x 或 x 的前驱 二分查找算法讲解 枚举查找即顺序查找, 实现原理是逐个比较数组 a[0:n-1] 中的元素,直到找到元素 x 或搜索整个数组后确定 x 不在…

linux网络服务学习(1):nfs

1.什么是nfs NFS:网络文件系统。 *让客户端通过网络访问服务器磁盘中的数据,是一种在linux系统间磁盘文件共享的方法。 *nfs客户端可以把远端nfs服务器的目录挂载到本地。 *nfs服务器一般用来共享视频、图片等静态数据。一般是作为被读取的对象&…

国内git最新版本下载链接2.44

git官网地址:Git - Downloading Package (git-scm.com) 蓝奏云: ​​​​​​gGit-2.44.0-64-bit.exe - 蓝奏云 git仓库地址:git/git: Git Source Code Mirror - This is a publish-only repository but pull requests can be turned into patches to the mailing list via …

算法笔记p251队列循环队列

目录 队列循环队列循环队列的定义初始化判空判满入队出队获取队列内元素的个数取队首元素取队尾元素 队列 队列是一种先进先出的数据结构,总是从队尾加入元素,从队首移除元素,满足先进先出的原则。队列的常用操作包括获取队列内元素的个数&a…

Typecho博客后台登陆界面美化

登录界面: 食用方法: 备份 admin 目录 压缩包内容上传到 admin 目录内。 结构:网站根目录 /admin/login.php 结构:网站根目录 /admin/style 修改 login.php 第35行,把“季春二九管理后台”替换成自己的信息 清理缓存,开始体验新的…

释放创造力,Nik Collection 6 by DxO 点亮你的视觉世界

在数字摄影时代,后期处理是提升摄影作品品质的重要环节。而Nik Collection 6 by DxO作为一套优秀的滤镜插件套装,不仅为摄影师提供了丰富的后期处理工具,更让他们能够释放无限的创造力,打造出惊艳的视觉作品。 Nik Collection 6 …

虚拟游戏理财 - 华为OD统一考试(C卷)

OD统一考试(C卷) 分值: 100分 题解: Java / Python / C++ 题目描述 在一款虚拟游戏中生活,你必须进行投资以增强在虚拟游戏中的资产以免被淘汰出局。 现有一家Bank,它提供有若干理财产品m,风险及投资回报不同,你有N (元)进行投资,能接受的总风,险值为X。 你要在可接…

WanAndroid(鸿蒙版)开发的第六篇

前言 DevEco Studio版本:4.0.0.600 WanAndroid的API链接:玩Android 开放API-玩Android - wanandroid.com 其他篇文章参考: 1、WanAndroid(鸿蒙版)开发的第一篇 2、WanAndroid(鸿蒙版)开发的第二篇 3、WanAndroid(鸿蒙版)开发的第三篇 …

Android Studio实现内容丰富的安卓视频管理平台

获取源码请点击文章末尾QQ名片联系,源码不免费,尊重创作,尊重劳动 项目编号081 1. 开发环境 android stuido 2.功能介绍 安卓端: 1.注册登录 2.本地视频 3.视频播放 4.收藏功能 5.网路视频 6.个人中心 7.我的收藏 8.浏览历史 3.系…