Java学习笔记21——使用JDBC访问MySQL数据库_java mysql 访问-程序员宅基地

技术标签: 学习  java  笔记  

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());

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/2303_79232676/article/details/136847772

智能推荐

稀疏编码的数学基础与理论分析-程序员宅基地

文章浏览阅读290次,点赞8次,收藏10次。1.背景介绍稀疏编码是一种用于处理稀疏数据的编码技术,其主要应用于信息传输、存储和处理等领域。稀疏数据是指数据中大部分元素为零或近似于零的数据,例如文本、图像、音频、视频等。稀疏编码的核心思想是将稀疏数据表示为非零元素和它们对应的位置信息,从而减少存储空间和计算复杂度。稀疏编码的研究起源于1990年代,随着大数据时代的到来,稀疏编码技术的应用范围和影响力不断扩大。目前,稀疏编码已经成为计算...

EasyGBS国标流媒体服务器GB28181国标方案安装使用文档-程序员宅基地

文章浏览阅读217次。EasyGBS - GB28181 国标方案安装使用文档下载安装包下载,正式使用需商业授权, 功能一致在线演示在线API架构图EasySIPCMSSIP 中心信令服务, 单节点, 自带一个 Redis Server, 随 EasySIPCMS 自启动, 不需要手动运行EasySIPSMSSIP 流媒体服务, 根..._easygbs-windows-2.6.0-23042316使用文档

【Web】记录巅峰极客2023 BabyURL题目复现——Jackson原生链_原生jackson 反序列化链子-程序员宅基地

文章浏览阅读1.2k次,点赞27次,收藏7次。2023巅峰极客 BabyURL之前AliyunCTF Bypassit I这题考查了这样一条链子:其实就是Jackson的原生反序列化利用今天复现的这题也是大同小异,一起来整一下。_原生jackson 反序列化链子

一文搞懂SpringCloud,详解干货,做好笔记_spring cloud-程序员宅基地

文章浏览阅读734次,点赞9次,收藏7次。微服务架构简单的说就是将单体应用进一步拆分,拆分成更小的服务,每个服务都是一个可以独立运行的项目。这么多小服务,如何管理他们?(服务治理 注册中心[服务注册 发现 剔除])这么多小服务,他们之间如何通讯?这么多小服务,客户端怎么访问他们?(网关)这么多小服务,一旦出现问题了,应该如何自处理?(容错)这么多小服务,一旦出现问题了,应该如何排错?(链路追踪)对于上面的问题,是任何一个微服务设计者都不能绕过去的,因此大部分的微服务产品都针对每一个问题提供了相应的组件来解决它们。_spring cloud

Js实现图片点击切换与轮播-程序员宅基地

文章浏览阅读5.9k次,点赞6次,收藏20次。Js实现图片点击切换与轮播图片点击切换<!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title></title> <script type="text/ja..._点击图片进行轮播图切换

tensorflow-gpu版本安装教程(过程详细)_tensorflow gpu版本安装-程序员宅基地

文章浏览阅读10w+次,点赞245次,收藏1.5k次。在开始安装前,如果你的电脑装过tensorflow,请先把他们卸载干净,包括依赖的包(tensorflow-estimator、tensorboard、tensorflow、keras-applications、keras-preprocessing),不然后续安装了tensorflow-gpu可能会出现找不到cuda的问题。cuda、cudnn。..._tensorflow gpu版本安装

随便推点

物联网时代 权限滥用漏洞的攻击及防御-程序员宅基地

文章浏览阅读243次。0x00 简介权限滥用漏洞一般归类于逻辑问题,是指服务端功能开放过多或权限限制不严格,导致攻击者可以通过直接或间接调用的方式达到攻击效果。随着物联网时代的到来,这种漏洞已经屡见不鲜,各种漏洞组合利用也是千奇百怪、五花八门,这里总结漏洞是为了更好地应对和预防,如有不妥之处还请业内人士多多指教。0x01 背景2014年4月,在比特币飞涨的时代某网站曾经..._使用物联网漏洞的使用者

Visual Odometry and Depth Calculation--Epipolar Geometry--Direct Method--PnP_normalized plane coordinates-程序员宅基地

文章浏览阅读786次。A. Epipolar geometry and triangulationThe epipolar geometry mainly adopts the feature point method, such as SIFT, SURF and ORB, etc. to obtain the feature points corresponding to two frames of images. As shown in Figure 1, let the first image be ​ and th_normalized plane coordinates

开放信息抽取(OIE)系统(三)-- 第二代开放信息抽取系统(人工规则, rule-based, 先抽取关系)_语义角色增强的关系抽取-程序员宅基地

文章浏览阅读708次,点赞2次,收藏3次。开放信息抽取(OIE)系统(三)-- 第二代开放信息抽取系统(人工规则, rule-based, 先关系再实体)一.第二代开放信息抽取系统背景​ 第一代开放信息抽取系统(Open Information Extraction, OIE, learning-based, 自学习, 先抽取实体)通常抽取大量冗余信息,为了消除这些冗余信息,诞生了第二代开放信息抽取系统。二.第二代开放信息抽取系统历史第二代开放信息抽取系统着眼于解决第一代系统的三大问题: 大量非信息性提取(即省略关键信息的提取)、_语义角色增强的关系抽取

10个顶尖响应式HTML5网页_html欢迎页面-程序员宅基地

文章浏览阅读1.1w次,点赞6次,收藏51次。快速完成网页设计,10个顶尖响应式HTML5网页模板助你一臂之力为了寻找一个优质的网页模板,网页设计师和开发者往往可能会花上大半天的时间。不过幸运的是,现在的网页设计师和开发人员已经开始共享HTML5,Bootstrap和CSS3中的免费网页模板资源。鉴于网站模板的灵活性和强大的功能,现在广大设计师和开发者对html5网站的实际需求日益增长。为了造福大众,Mockplus的小伙伴整理了2018年最..._html欢迎页面

计算机二级 考试科目,2018全国计算机等级考试调整,一、二级都增加了考试科目...-程序员宅基地

文章浏览阅读282次。原标题:2018全国计算机等级考试调整,一、二级都增加了考试科目全国计算机等级考试将于9月15-17日举行。在备考的最后冲刺阶段,小编为大家整理了今年新公布的全国计算机等级考试调整方案,希望对备考的小伙伴有所帮助,快随小编往下看吧!从2018年3月开始,全国计算机等级考试实施2018版考试大纲,并按新体系开考各个考试级别。具体调整内容如下:一、考试级别及科目1.一级新增“网络安全素质教育”科目(代..._计算机二级增报科目什么意思

conan简单使用_apt install conan-程序员宅基地

文章浏览阅读240次。conan简单使用。_apt install conan