JDBC
我的MySQL的root密码: root
11.1 什么是JDBC
11.1.1 JDBC概述
JDBC的全称是Java数据库连接(Java Database Connectivity),它是一套用于执行SQL语句的JavaAPI。应用程序可通过这套JavaAPI连接到关系数据库,并使用SQL语句完成对数据库中数据的查询、新增、更新和删除等操作。不同的数据库(如MySQL、Oracle等)处理数据的方式是不同的,如果直接使用数据库厂商提供的访问接口操作数据库,应用程序的可移植性就会变得很差。例如,用户在当前程序中使用的是MySQL提供的接口操作数据库,如果换成Oracle数据库,则需要重新使用Oracle数据库提供的接口,这样代码的改动量会非常大。如果使用JDBC,上述问题就不复存在了,因为JDBC要求各个数据库厂商按照统一的规范提供数据库驱动程序,在程序中由JDBC和具体的数据库驱动程序联系,用户不必直接与底层的数据库交互,使得代码的通用性更强。
11.1.2 JDBC驱动程序
JDBC本身提供的是一套数据库操作标准,而JDBC中提供的这些标准又需要由各个数据库厂商实现,每一个数据库厂商都会为其数据库产品提供一个JDBC驱动程序。目前比较常见的JDBC驱动程序可以分为以下4类。
1.JDBC-ODBC桥驱动程序
JDBC-ODBC桥驱动程序由Sun公司开发,是JDK提供的数据库操作标准API,这种类型的驱动程序实际是把所有JDBC的调用传递给ODBC(Open Database Connectivity,开放数据库连接),再由ODBC调用本地数据库驱动程序代码,操作数据库中的数据。通过JDBC-ODBC桥驱动程序操作数据库的方式如图11-2所示。由于JDBC-ODBC桥驱动程序经过几次中间调用,所以执行效率比较低。
2.本地API驱动程序
本地API驱动程序直接将JDBCAPI映射成数据库特定的客户端API。这种驱动程序包含特定数据库的本地API,通过它可以访问数据库的客户端。通过本地API驱动程序操作数据库的方式如图11-3所示。通过本地API驱动程序访问数据库减少了ODBC的调用环节,提高了数据库访问的效率。
3.网络协议驱动程序
网络协议驱动是用纯Java语言编写的。JDBC把对数据库的访问请求传递给网络上的中间件服务器;中间件服务器先把请求转换为数据库通信协议请求,然后再与数据库进行交互。使用这种类型的JDBC驱动程序的Java应用程序可以与服务器端完全分离,具有很大的灵活性。通过网络协议驱动程序操作数据库的方式如图11-4所示。
4.本地协议驱动程序
本地协议驱动程序是用使用纯Java语言编写的。本地协议驱动程序通常是由数据库厂商直接提供的JAR包。本地协议驱动程序直接将JDBC调用转换为数据库特定的网络通信协议,然后与数据库进行交互。通过本地协议驱动程序操作数据库的方式如图11-5所示。
在上述4种类型中,JDBC-ODBC桥驱动程序由于执行效率不高,更适合作为开发应用时的一种过渡方案;如果是在内联网(Intranet)中的应用,可以考虑本地API驱动程序;如果是基于互联网(Internet)并且需要同时连接多个不同种类的数据库、并发连接要求高的应用,可以考虑网络协议驱动程序;如果是基于互联网(Internet)但连接单一数据库的应用,可以考虑本地协议驱动程序。本章将基于驱动程序类型对JDBC进行讲解。
11.2 JDBC的常用API
1.Driver接口
Driver接口是所有JDBC驱动程序必须实现的接口,该接口专门提供给数据库厂商使用。需要注意的是,在编写Java应用程序时,必须把使用的数据库驱动程序(这里指MySQL驱动程序的JAR包)或类库加载到项目的classpath中。
2.DriverManager类
使用JDBC连接数据库,需要用到DriverManager类,它用于加载JDBC驱动程序并且创建Java应用程序与数据库的连接。在DriverManager类中,定义了两个重要的静态方法。
方法名称 | 功能描述 |
---|---|
registerDriver(Driver driver) | 该方法用于向DriverManager类注册给定的JDBC驱动程序 |
getConnection(String url,String user,String password) | 该方法用于建立和数据库的连接,并返回表示连接的Connection对象 |
3.Connection接口
DriverManager类的getConnection()方法返回一个Connection对象,它是表示数据库连接的对象,只有获得该对象,才能访问并操作数据库。一个应用程序可与单个数据库建立一个或多个连接,也可以与多个数据库建立连接。
方法声明 | 功能描述 |
---|---|
createStatement() | 用于创建一个Statement对象,该对象可以将SQL语句发送到数据库 |
prepareStatement(String sql) | 用于创建一个PreparedStatement对象,该对象可以将参数化的动态SQL语句发送到数据库 |
prepareCall(String sql) | 用于创建一个CallableStatement对象以调用数据库的存储过程 |
isReadOnly() | 用于查看当前Connection对象的读写模式是否为只读模式 |
setReadOnly() | 用于设置当前Connection对象的读写模式,默认为非只读模式 |
commit() | 使所有上一次提交/回滚后进行的更改成为持久更改,并释放当前Connection对象持有的所有数据库锁 |
setAutoCommit(boolean autoCommit) | 设置是否关闭自动提交模式 |
roolback() | 用于取消在当前事务中进行的所有更改,并释放当前Connection对象持有的所有数据库锁 |
close() | 用于立即释放当前Connection对象的数据库和JDBC资源,而不是等它们被自动释放 |
isClose() | 用于判断当前Connection对象是否已被自动关闭 |
4.Statement接口Statement接口用于执行静态的SQL语句,并返回一个结果对象。Statement接口对象可以通过Connection实例的createStatement()方法获得,该对象会把静态的SQL语句发送到数据库中编译执行,然后返回数据库的处理结果。Statement接口提供了3个执行SQL语句的常用方法。
方法声明 | 功能描述 |
---|---|
execute(String sql) | 用于执行各种SQL语句。该方法返回一个boolean类型的值,如果返回值为true,表示执行的SQL语句有查询结果,可以通过Statement接口的getResultSet()方法获得查询结果 |
executeUpdate(String sql) | 用于执行SQL中的insert、update和delete语句。该方法返回一个int类型的值,表示数据库中受该SQL语句影响的记录条数 |
executeQuery(String sql) | 用于执行SQL中的select语句。该方法返回一个表示查询结果的ResultSet对象 |
5.PreparedStatement接口
Statement接口封装了JDBC执行SQL语句的方法,可以完成Java程序执行SQL语句的操作。然而在实际开发过程中往往需要将程序中的变量作为SQL语句的查询条件,而使用Statement接口操作这些SQL语句过于烦琐,并且存在安全方面的问题。针对这一问题,JDBC API提供了扩展的PreparedStatement接口。PreparedStatement是Statement的子接口,用于执行预编译的SQL语句。PreparedStatement接口扩展了带参数的SQL语句的执行操作,该接口中的SQL语句可以使用占位符“?”代替参数,然后通过setter方法为SQL语句的参数赋值。
方法声明 | 功能描述 |
---|---|
executeUpdate() | 在PreparedStatement对象中执行SQL语句。SQL语句必须是一个DML语句或者是无返回内容的SQL语句(如DDL语句) |
executeQuery() | 在PreparedStatement对象中执行SQL查询。该方法返回的是ResultSet对象 |
setInt(int Index,int x) | 将指定位置的参数设置为指定的int类型的值 |
setFloat(int index,float f) | 将指定位置的参数设置为float类型的值 |
setLong(int index,long l) | 将指定位置的参数设置为long类型的值 |
setDouble(int index,double d) | 将指定位置的参数设置为double类型的值 |
setBoolean(int index,boolean b) | 将指定位置的参数设置为boolean类型的值 |
void setString(int Index,String x) | 将指定位置的参数设置为指定的String类型的值 |
在表11-4中,DML(数据操纵语言)语句指的是操作数据库、表、列等的语句,使用的关键字为CREATE、ALTER、DROP。DDL(数据定义语言)语句指的是对表中的数据进行增、删、改操作的语句,使用的关键字为INSERT、UPDATE、DELETE。在为SQL语句中的参数赋值时,可以通过输人参数与SQL类型相匹配的setXxx()方法赋值。例如,字段的数据类型为int或Integer,那么可以使用setInt()方法或setObject()方法设置输人参数,具体示例如下:
String sql="INSERT INTO users(id,name,email) VALUES(?,?,?)";
Preparedstatement preStmt=conn.prepareStatement(sql);
preStmt.setInt(1,1);//使用参数与SQL类型相匹配的方法
preStmt.setString(2,"zhangsan");//使用参数与SQL类型相匹配的方法
preStmt.setobject(3,"[email protected]");//使用setObject()方法设置参数
preStmt.executeUpdate();
6.ResultSet接口
ResultSet接口用于保存JDBC执行查询时返回的结果集,该结果集封装在一个逻辑表格中。在ResultSet接口内部有一个指向表格数据行的游标(或指针),ResultSet对象初始化时游标在表格的第一行之前,调用next()方法可以将游标移动到下一行。如果下一行没有数据,则next()方法返回false。在应用程序中经常使用next()方法作为while循环的条件来迭代结果集。
方法声明 | 功能描述 |
---|---|
getString(int columnIndex) | 用于获取指定字段的String类型的值,参数columnIndex代表字段的索引 |
getString(String columnName) | 用于获取指定字段的String类型的值,参数columnName代表字段的名称 |
getInt(int columnIndex) | 用于获取指定字段的int类型的值 |
getInt(String columnName) | 用于获取指定字段的int类型的值 |
absolute(int row) | 将游标移动到结果集的第row条记录 |
relative(int row) | 按相对行数(正或负)移动游标 |
previous() | 将游标从结果集的当前位置移动到上一行 |
next() | 将游标从结果集的当前位置移动到下一行 |
beforeFirst() | 将游标移动到结果集的开头(第一行之前) |
isBeforeFirst() | 判断游标是否位于结果集的开头(第一行之前) |
afterLast() | 将游标指针移动到结果集的末尾(最后一行之后) |
isAfterLast() | 判断游标是否位于结果集的末尾(最后一行之后) |
first() | 将游标移动到结果集的第一行 |
isFirst() | 判断游标是否位于结果集的第一行 |
last() | 将游标移动到结果集的最后一行 |
getRow() | 返回当前记录的行号 |
getStatement() | 返回生成结果集的Statement对象 |
close() | 释放当前结果集的数据库和JDBC资源 |
11.3 JDBC编程
11.3.1 JDBC 编程步骤
通常情况下,使用JDBCAPI实现JDBC程序时,首先需要加载并注册数据库驱动程序,其次使用DriverManager类调用getConnection()方法获取表示数据连接的Connection对象,最后通过Connection对象调用相应方法获取Statement对象,通过Statement对象执行SQL语句。执行SQL语句之后,如果数据库有返回结果,则将结果封装为ResultSet对象返回。
1 加载并注册数据库驱动程序
在连接数据库之前,要加载数据库的驱动程序到Java虚拟机。加载数据库驱动程序操作可以通过java.lang.Class类的静态方法forName(StringclassName)或DriverManager类的静态方法registerDriver(Driverdriver)实现,具体示例如下:
DiverManager.registerDiver(Driver Driver)
或者
Class.forName("DriverName");
调用registDriver()方法时,实际上创建了两个Driver对象,对于只需加载驱动程序类来讲有些浪费资源;使用forName()方法,驱动程序类的名称是以字符串的形式填写的,使用时可以把该驱动程序类的名称放到配置文件中,如果需要切换驱动程序类会非常方便。所以在实际开发中,常调用forName()方法注册数据库驱动程序。forName()方法中的参数DriverName表示数据库的驱动程序类。以MySQL数据库为例,MySQL驱动程序类在MySQL6.0.2版本之前是com.mysql.jdbc.Driver,加载示例代码如下:
Class.forName("com.mysql.jdbc.Driver");
而在MySQL6.0.2版本之后,MySQL驱动程序类是com.mysql.cj.jdbc.Driver,加载示例代码如下:
Class.forName("com.mysql.cj.jdbc.Driver");
在实际加载时,用户需要根据数据库版本选择对应的驱动程序类
2 通过DriverManager类获取数据库连接
DriverManager类的getConnection()方法用于获取JDBC驱动程序到数据库的连接,通过DriverManager类获取数据库连接的具体方式如下:
Connection conn = DriverManager.getConnection(String url,String user,String pwd);
从上述代码可以看出,getConnection()方法有3个参数,分别表示连接数据库的URL、登录数据库的用户名和登录数据库的密码。以MySQL数据库为例,数据库地址的书写格式如下:
jdbc:mysql://hostname:port/databasename
在上面的代码中,“jdbc:mysql:”是固定的写法,后面的hostname指的是主机的名称(如果数据库在本机中,hostname可以为localhost或127.0.0.1;如果要连接的数据库在其他计算机中,hostname可以是连接计算机的IP地址),port指的是连接数据库的端口号(MySQL端口号默认为3306),databasename指的是MySQL中相应数据库的名称。
3.通过Connection对象获取Statement对象
获取Connection对象之后,还必须创建Statement对象,将各种SQL语句发送到连接的数据库中执行。如果把Connection对象看作一条连接程序和数据库的索道,那么Statement对象就可以看作索道上的一辆缆车,它为数据库传输SQL语句,并返回执行结果。Connection创建Statement对象的方法有以下3个:
(1)createStatement():创建基本的Statement对象。
(2)prepareStatement():根据传递的SQL语句创建PreparedStatement对象。
(3)prepareCall():根据传人的SQL语句创建CallableStatement对象。
以创建基本的Statement对象为例,创建方式如下:
Statement stmt=conn.createStatement();
4.使用Statement执行SQL语句
创建了Statement对象后,就可以通过该对象执行SQL语句。如果SQL语句运行后产生了结果集,Statement对象会将结果集封装成ResultSet对象并返回。Statement有以下3个执行SQL语句的方法:
- execute():可以执行任何SQL语句。
- executeQuery():通常执行查询语句,执行后返回代表结果集的ResultSet对象。
- executeUpdate():主要用于执行DML语句和DDL语句。执行DML语句(如INSERT、UPDATE或DELETE)返回受SQL语句影响的行数;执行DDL语句返回0。以executeQuery()方法为例,调用形式如下:
//创建SQL语句
String sql="SELECT name from users WHERE id=1";
//执行SQL语句,获取结果集
ResultSet rs=stmt.executeQuery(sql);
需要注意的是,在调用executeQuery()方法和executeUpdate()方法时,如果作为参数的SQL语句有多行(多条SQL语句),将出现编译错误,因此在构造SQL语句时,如果SQL语句有多行,需要将多行SQL语句加上双引号并使用十号连接起来。例如下面的SQL语句:
String sql="INSERT INTO users(NAME,PASSWORD,email,birthday)"
+VALUES('zhangs','123456','zs@sina.com','1980-12-04');"
"ResultSetrs=stmt.executeQuery(sgl);
5.操作结果集
如果执行的SQL语句是查询语句,执行结果将封装成一个ResultSet对象,该对象保存了SQL语句查询的结果。程序可以通过操作该ResultSet对象取出查询结果。
6.关闭连接并释放资源
每次操作数据库结束后都要关闭数据库连接并释放资源,以防止系统资源浪费。资源的关闭顺序和声明顺序相反,关闭顺序依次为关闭ResultSet对象、关闭Statement对象、关闭Connection对象。为了保证在异常情况下也能关闭资源,通常在try".catch语句的finally代码块中统一关闭资源。至此,JDBC程序的大致实现步骤已经讲解完成。
11.3.2 实现第一个JDBC程序
(1)搭建数据库环境。在MySQL中创建名称为jdbc的数据库,然后在jdbc数据库中创建users表,创建数据库和表的SQL语句如下:
CREATE DATABASE jdbc;
USE jdbc;
CREATE TABLE users(
id INT PRIMARY KEY AUTO_INCREMENT,
name VARCHAR(40),
password VARCHAR(40),
email VARCHAR(60),
birthday DATE
);
jdbc数据库和users表创建成功后,再向users表中插人3条数据,SQL语句如下:
insert into users(NAME,PASSWORD,email,birthday) values('zhangs','123456','[email protected]','1980-12-04'), ('1isi','123456','[email protected]','1981-12-04'),('wangwu','123456','[email protected]','1979-12-04');
为了查看数据是否添加成功,使用SELECT语句查询users表中的数据,执行结果如图11-7所示。
(2)创建项目环境,导人数据库驱动程序。在IDEA中新建名称为chapterl1的Java项目,右击项目名称,在弹出的快捷菜单中选择New→Directory命令,在弹出的对话框中将该目录命名为lib,项目根目录中就会出现一个名称为lib的目录。将下载好的MySQL数据库驱动程序文件mysql-connector-java-8.0.15.jar复制到项目的lib目录中,并把JAR包添加到项目里。在IDEA菜单栏中选择File→ProjectStructure→Modules→Dependencies命令,单击最右侧加号后选择第一项:“JARsordirectories"”,在弹出的对话框中选择下载好的JAR包并确认。最后可以看到mysql-connector-java-8.0.15.jar已添加到IDEA的依赖项中。成功添加MySQL的JAR包的界面如图所示。
将mysql-connector-java-8.0.15.jar添加到依赖项之后,单击Apply按钮,再单击OK按钮,可以看到在ExternalLibraries下已经存在刚刚添加的JAR包。至此,JAR包添加成功。加人数据库驱动程序后的项目结构如图所示。
(3)编写JDBC程序。在项目chapterl1的src目录下,新建名称为com.itheima.jdbc.example的包,在该包中创建Example01类,该类用于读取数据库中的users表,并将结果输出到控制台。Example01类的实现如文件所示。
import java.sql.*;
public class Example01 {
public static void main(String[] args) throws Exception {
Statement stmt = null;
ResultSet rs = null;
Connection conn = null;
try {
// 1. 注册数据库的驱动
Class.forName("com.mysql.cj.jdbc.Driver");
// 2.通过DriverManager获取数据库连接
String url =
"jdbc:mysql://localhost:3306/jdbc?serverTimezone=GMT%2B8&useSSL=false";
String username = "root"; //数据库用户名
String password = "root"; //数据库密码
conn = DriverManager.getConnection(url, username, password);
// 3.通过Connection对象获取Statement对象
stmt = conn.createStatement();
// 4.使用Statement执行SQL语句。
String sql = "select * from users";
rs = stmt.executeQuery(sql);
// 5. 操作ResultSet结果集
System.out.println("id | name | password| email | birthday");
while (rs.next()) {
int id = rs.getInt("id"); // 通过列名获取指定字段的值
String name = rs.getString("name");
String psw = rs.getString("password");
String email = rs.getString("email");
Date birthday = rs.getDate("birthday");
System.out.println(id + " | " + name + " | " + psw + " | " + email + " | " + birthday);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
// 6.回收数据库资源
if (rs != null) {
try {
rs.close();
} catch (SQLException e) {
e.printStackTrace();
}
rs = null;
}
if (stmt != null) {
try {
stmt.close();
} catch (SQLException e) {
e.printStackTrace();
}
stmt = null;
}
if (conn != null) {
try {
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
conn = null;
}
}
}
}
在文件中,第9行代码通过Class的forName()方法注册了MySQL数据库驱动程序。
第11~16行代码通过DriverManager的getConnection()方法获取数据库的连接。
第18行代码通过Connection对象获取Statement对象。
第20、21行代码调用Statement对象的executeQuery()方法执行SQL查询语句。第23~33行代码使用ResultSet对象操作结果集,并用while循环获取数据库的所有数据。
第38~61行代码依次关闭ResultSet对象、Statement对象和Connection对象并释放资源。
运行结果所示。users表中的数据已输出控制台。至此,第一个JDBC程序成功实现。在实现这个JDBC程序时,有以下3点需要注意。
(1)注册驱动程序。虽然使用DriverManager.registerDriver(newcom.mysql.cj.jdbc.Driver())方法也可以完成注册,但这种方式会使数据库驱动程序被注册两次。因为在Driver类的源码中,已经在静态代码块中完成了数据库驱动程序的注册。为了避免数据库驱动程序被重复注册,在程序中使用Class.forName()方法加载驱动程序类即可。
(2)释放资源。由于数据库资源非常宝贵,数据库允许的并发访问连接数量有限,因此,当数据库资源使用完毕后,一定要释放资源。为了保证资源被释放,在Java程序中应该将释放资源的操作放在finally代码块中。
(3)获取数据库连接。在MySQL8.0及以上版本中获取数据库连接时需要设置时区为北京时间(serverTimezone=GMT%2B8),因为安装数据库时默认为美国时间。如果不设置时区为北京时间,系统会报告MySQL设置时区与当前系统时区不符的错误
更换URL地址
url=jdbc:mysql://127.0.0.1:3306/jdbc?characterEncoding=utf8&useSSL=true
本章主要讲解了JDBC的基本知识,包括什么是JDBC、JDBC的主要类和接口、JDBC的人门程序。通过本章的学习,读者可以了解什么是JDBC,熟悉JDBC的主要类和接口,掌握如何使用JDBC操作数据库中的数据。