第十四讲 JDBC数据库

news2025/1/26 15:41:10

1. 什么是JDBC

JDBC(Java Database Connectivity,Java数据库连接),它是一套用于执行SQL语句的Java API。应用程序可通过这套API连接到关系型数据库,并使用SQL语句来完成对数据库中数据的查询、新增、更新和删除等操作。

应用程序使用JDBC访问数据库的方式如图1所示。

图1 应用程序通过JDBC访问数据库方式

2. JDBC常用API

1Driver接口

Driver接口是所有JDBC驱动程序必须实现的接口,该接口专门提供给数据库厂商使用。

注意:在编写JDBC程序时,必须要把所使用的数据库驱动程序或类库加载到项目的classpath中(这里指数据库的驱动JAR包)。

(2)DriverManager类

DriverManager类用于加载JDBC驱动并且创建与数据库的连接。其主要方法如表1所示。

表1 DriverManager类常用方法

方法声明

功能描述

static synchronized void registerDriver(Driver driver)

该方法用于向DriverManager中注册给定的JDBC驱动程序

static Connection getConnection(String url,String user,String pwd)

该方法用于建立和数据库的连接,并返回表示连接的Connection对象

说明:在实际开发中,通常不使用registerDriver(Driverdriver)注册驱动。因为JDBC驱动类Driver中有一段静态代码块,是向DriverManager注册一个Driver实例,当再次执行registerDriver(newDriver()),相当于实例化了两个Driver对象,因此在加载数据库驱动时通常使用Class类的静态方法forName()来实现。

(3) Connection接口

Connection接口代表Java程序和数据库的连接对象,只有获得该连接对象后,才能访问数据库,并操作数据表。Connection接口常用方法如表2所示。

表2 Connection接口

方法声明

功能描述

Statement createStatement()

该方法用于返回一个向数据库发送语句的Statement对象

PreparedStatement prepareStatement(String sql)

该方法用于返回一个PreparedStatement对象,该对象用于向数据库发送参数化的SQL语句

CallableStatement prepareCall(String sql)

该方法用于返回一个CallableStatement对象,该对象用于调用数据库中的存储过程

(4) Statement接口

Statement是Java执行数据库操作的一个重要接口,它用于执行静态的SQL语句,并返回一个结果对象。

说明:Statement接口对象可以通过Connection实例的createStatement()方法获得,然后返回数据库的处理结果。Statement接口常用方法如表3所示。

表3 Statement常用方法

方法声明

功能描述

boolean execute(String sql)

用于执行各种SQL语句,返回一个boolean类型的值,如果为true,表示所执行的SQL语句有查询结果,可通过Statement的getResultSet()方法获得查询结果

int executeUpdate(String sql)

用于执行SQL中的insert、update和delete语句。该方法返回一个int类型的值,表示数据库中受该SQL语句影响的记录条数

ResultSet 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语句可以使用占位符“?”来代替其参数,然后通过setXxx()方法为SQL语句的参数赋值。

PreparedStatement接口常用方法如表4所示。

表4 PreparedStatement接口常用方法

方法声明

功能描述

int executeUpdate()

在此PreparedStatement对象中执行 SQL 语句,该语句必须是一个DML语句或者是无返回内容的SQL 语句,如 DDL 语句

ResultSet executeQuery()

在此PreparedStatement对象中执行 SQL 查询,该方法返回的是ResultSet对象

void setInt(int parameterIndex, int x)

将指定参数设置为给定的int值

void setFloat(int parameterIndex, float x)

将指定参数设置为给定的float值

void setString(int parameterIndex, String x)

将指定参数设置为给定的String值

void setDate(int parameterIndex, Date x)

将指定参数设置为给定的Date值

void addBatch()

将一组参数添加到此PreparedStatement对象的批处理命令中

void setCharacterStream(int parameterIndex,  java.io.Reader reader,  int length)

将指定的输入流写入数据库的文本字段

void setBinaryStream(int parameterIndex, java.io.InputStream x, int length)

将二进制的输入流数据写入到二进制字段中

为SQL语句参数赋值时,可以通过输入参数与SQL类型相匹配的setXxx()方法。例如字段的数据类型为int或Integer,那么应该使用setInt()方法,也可以通过setObject()方法设置多种类型的输入参数。

    // 假设users表中字段id、name、email类型分别是int、varchar、varchar

    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, "zs@sina.com");     //使用setObject()方法设置参数

    preStmt.executeUpdate();

(6)ResultSet接口

ResultSet接口用于保存JDBC执行查询时返回的结果集,该结果集封装在一个逻辑表格中。在ResultSet接口内部有一个指向表格数据行的游标(或指针),ResultSet对象初始化时,游标在表格的第一行之前,调用next()方法可将游标移动到下一行。如果下一行没有数据,则返回false。在程序中经常使用next()方法作为while循环的条件来迭代ResultSet结果集。ResultSet接口常用方法如表5所示。

表5 ResultSet接口常用方法

方法声明

功能描述

String getString(int

 columnIndex)

用于获取指定字段的String类型的值,参数columnIndex代表字段的索引

String getString(String

 columnName)

用于获取指定字段的String类型的值,参数columnName代表字段的名称

int getInt(int columnIndex)

用于获取指定字段的int类型的值,参数columnIndex代表字段的索引

int getInt(String columnName)

用于获取指定字段的int类型的值,参数columnName代表字段的名称

Date getDate(int columnIndex)

用于获取指定字段的Date类型的值,参数columnIndex代表字段的索引

Date getDate(String

 columnName)

用于获取指定字段的Date类型的值,参数columnName代表字段的名称

boolean next()

将游标从当前位置向下移一行

boolean absolute(int row)

将游标移动到此 ResultSet 对象的指定行

void afterLast()

将游标移动到此 ResultSet 对象的末尾,即最后一行之后

void beforeFirst()

将游标移动到此 ResultSet 对象的开头,即第一行之前

boolean previous()

将游标移动到此 ResultSet 对象的上一行

boolean last()

将游标移动到此 ResultSet 对象的最后一行

ResultSet接口中定义了大量的getXxx()方法,而采用哪种getXxx()方法取决于字段的数据类型。

程序既可以通过字段的名称来获取指定数据,也可以通过字段的索引来获取指定的数据,字段的索引是从1开始编号的。

例如,假设数据表的第1列字段名为id,字段类型为int,那么既可以使用getInt("id")获取该列的值,也可以使用getInt(1)获取该列的值。

3. JDBC编程的基本步骤

通常情况下,JDBC编程可分为如下几个基本步骤。

(1)加载数据库驱动

 加载数据库驱动通常使用Class类的静态方法forName()来实现,使用格式:

Class.forName(DriverName);

上述代码中,DriverName代表的是数据库驱动类所对应的字符串。例如,要加载MySQL数据库的驱动可以采用如下代码。

Class.forName("com.mysql.jdbc.Driver");

(2)通过DriverManager获取数据库连接

DriverManager提供了一个getConnection()方法来获取数据库连接,使用格式如下。

Connection conn = DriverManager.getConnection(String url, String user, String pwd);

   参数说明:

url:表示连接数据库的URL。

user:表示登录数据库的用户名。

pwd:表示登录数据库的密码。

用户名和密码通常由数据库管理员设置,而连接数据库的URL则有固定格式。如MySQL数据库的URL地址为。

jdbc:mysql://hostname:prot/databasename

上述代码中,jdbc:mysql:是固定写法,mysql指的是MySQL数据库;hostname指的是MySQL数据库所在的主机名或IP地址(例如数据库在本机上,hostname可以是localhost或127.0.0.1);port指的是连接数据库的端口号(默认为3306);databasename指的是要操作的数据库。

(3)通过Connection对象获取Statement对象

Connection创建Statement的方式有3种。

①createStatement():创建基本的Statement对象。

②prepareStatement(String sql):根据传递的SQL语句创建PreparedStatement对象。

③prepareCall(Stringsql):根据传入的SQL语句创建CallableStatement对象。

例创建基本的Statement对象,其代码如下所示。

Statement stmt = conn.createStatement();

(4)使用Statement执行SQL语句

可通过如下3种不同的方式来执行SQL语句。

①execute(String sql):用于执行任意的SQL语句。

②executeQuery(String sql):用于执行查询语句,返回ResultSet结果集对象。

③executeUpdate(String sql):主要用于执行DML(数据操作语言)和DDL(数据定义语言)语句。执行DML语句(INSERT、UPDATE或DELETE)时,会返回受SQL语句影响的行数,执行DDL(CREATE、ALTER)语句返回0。

例执行查询SQL,获取结果集:

// 执行SQL语句,获取结果集ResultSet

ResultSet rs = stmt.executeQuery(sql);

(5)操作ResultSet结果集

如果执行的SQL语句是查询语句,执行结果将返回一个ResultSet对象,该对象里保存了SQL语句查询的结果。程序可以通过操作该ResultSet对象来取出查询结果。

(6)关闭连接,释放资源

每次操作数据库结束后都要关闭数据库连接,释放资源,以重复利用资源。通常资源的关闭顺序与打开顺序相反,顺序是ResultSet、Statement(或PreparedStatement)和Connection。为了保证在异常情况下也能关闭资源,需要在try...catch的finally代码块中统一关闭资源。

4. JDBC编程示例

(1)添加mysql驱动

①在项目中新建一lib目录,将mysql的驱动程序复制到此目录中。

②选择菜单【file】-【project structure】,左侧选择“modules”,右侧中间选择“Dependencies”,然后点击右上角的“+”号,选择“1 jars or directories…”,如下图所示。

③在弹出的对话框中选择项目中lib目录下的mysql驱动,然后点击”ok”返回。

④此时在Dependencies中会多了刚才选中的mysql驱动,然后点击“ok”即可。

(2)数据库操作

假设要操作的MySQL数据库名为db_student,db_student数据库中有一个表为tb_stud,表中内容如下所示。

【例10-1】读取表中内容并显示

import java.sql.*;


public class JDBCShow {

    public static void main(String[] args) throws SQLException {

        Connection conn = null;

        Statement stmt = null;

        ResultSet rs = null;

        try{

            Class.forName("com.mysql.jdbc.Driver");

            String url = "jdbc:mysql://localhost:3306/db_student?characterEncoding=utf8";

            String username="root";

            String password = "";

            conn = DriverManager.getConnection(url,username,password);

            stmt = conn.createStatement();

            rs = stmt.executeQuery("select * from tb_stud");

            System.out.println("no\t\tname\tmath\tchinese\tenglish");

            while(rs.next()){

                String no = rs.getString("no");

                String name = rs.getString("name");

                int math = rs.getInt("math");

                int chinese = rs.getInt("chinese");

                int english = rs.getInt("english");

                System.out.println(no + "\t" +name + "\t" + math + "\t\t" + chinese+"\t\t"+ english);

            }

        }catch(Exception e){

            e.printStackTrace();

        }finally{

            if(rs!=null) {

                rs.close();

            }

            if(stmt!=null){

                stmt.close();

            }

            if (conn!=null){

                conn.close();

            }

        }

    }

}

运行结果如下图所示。

【例10-2】先往表中插入记录,然后读取表中记录显示

import java.sql.*;


public class JDBCInsertShow {

    public static void main(String[] args) throws SQLException {

        Connection conn = null;

        Statement stmt = null;

        ResultSet rs = null;

        try{

            Class.forName("com.mysql.jdbc.Driver");

            String url = "jdbc:mysql://localhost:3306/db_student?characterEncoding=utf8";

            String username="root";

            String password = "";

            conn = DriverManager.getConnection(url,username,password);

            stmt = conn.createStatement();

            String sql = "insert into tb_stud(no,name,math,chinese,english) values(?,?,?,?,?)";

            PreparedStatement preStmt =conn.prepareStatement(sql);

            preStmt.setString(1,"1005");

            preStmt.setString(2,"张小");

            preStmt.setInt(3,80);

            preStmt.setInt(4,99);

            preStmt.setInt(5,90);

            preStmt.executeUpdate();

            rs = stmt.executeQuery("select * from tb_stud");

            System.out.println("no\t\tname\tmath\tchinese\tenglish");

            while(rs.next()){

                String no = rs.getString("no");

                String name = rs.getString("name");

                int math = rs.getInt("math");

                int chinese = rs.getInt("chinese");

                int english = rs.getInt("english");

                System.out.println(no + "\t" +name + "\t" + math + "\t\t" + chinese+"\t\t"+ english);

            }

        }catch(Exception e){

            e.printStackTrace();

        }finally{

            if(rs!=null) {

                rs.close();

            }

            if(stmt!=null){

                stmt.close();

            }

            if (conn!=null){

                conn.close();

            }

        }

    }

}

运行结果如下图所示。

此时数据库表中内容如下所示。

【例10-3】记录的增、删、改、显示操作

import java.sql.*;


public class JDBCOperator {

    public static void main(String[] args) throws SQLException {

        Connection conn = null;

        Statement stmt = null;

        ResultSet rs = null;

        try{

            Class.forName("com.mysql.jdbc.Driver");

            String url = "jdbc:mysql://localhost:3306/db_student?characterEncoding=utf8";

            String username = "root";

            String pw = "";

            conn = DriverManager.getConnection(url,username,pw);

            stmt = conn.createStatement();

            String sql = "update tb_stud set name='王明' where id=1";   //记录的修改

            stmt.executeUpdate(sql);

            sql = "insert into tb_stud values(null,'1004','张朋',70,70,70)"; //记录的添加

            stmt.executeUpdate(sql);

            sql = "delete from tb_stud where name='李军'";   //记录的删除

            stmt.executeUpdate(sql);

            rs = stmt.executeQuery("select * from tb_stud");

            System.out.println("no\t\tname\tmath\tchinese\tenglish");

            while(rs.next()){

                String no = rs.getString("no");

                String name = rs.getString("name");

                int math = rs.getInt("math");

                int chinese = rs.getInt("chinese");

                int english = rs.getInt("english");

                System.out.println(no + "\t" +name + "\t" + math + "\t\t" + chinese+"\t\t"+ english);

            }

        }catch(Exception e){

            e.printStackTrace();

}        }finally{

            if (rs!=null){

                rs.close();

            }

            if (stmt!=null) {

                stmt.close();

            }

            if (conn!=null) {

                conn.close();

            }

        }

    }

运行结果如下图所示。

此时表中内容如下所示。

 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java面试题2025-Mysql

1.什么是BufferPool? Buffer Pool基本概念 Buffer Pool:缓冲池,简称BP。其作用是用来缓存表数据与索引数据,减少磁盘IO操作,提升效率。 Buffer Pool由缓存数据页(Page) 和 对缓存数据页进行描述的控制块 组成, 控制…

Golang:使用DuckDB查询Parquet文件数据

本文介绍DuckDB查询Parquet文件的典型应用场景,掌握DuckDB会让你的产品分析能力更强,相反系统运营成本相对较低。为了示例完整,我也提供了如何使用Python导出MongoDB数据。 Apache Parquet文件格式在存储和传输大型数据集方面变得非常流行。最…

Linux网络之TCP

Socket编程--TCP TCP与UDP协议使用的套接字接口比较相似, 但TCP需要使用的接口更多, 细节也会更多. 接口 socket和bind不仅udp需要用到, tcp也需要. 此外还要用到三个函数: 服务端 1. int listen(int sockfd, int backlog); 头文件#include <sys/socket.h> 功能: …

工业相机 SDK 二次开发-Halcon 插件

本文介绍了 Halcon 连接相机时插件的使用。通过本套插件可连接海康 的工业相机。 一. 环境配置 1. 拷贝动态库 在 用 户 安 装 MVS 目 录 下 按 照 如 下 路 径 Development\ThirdPartyPlatformAdapter 找到目录为 HalconHDevelop 的文 件夹&#xff0c;根据 Halcon 版本找到对…

axios架构设计和原理

1. 前端请求方式 原生XHR&#xff1a;最初&#xff0c;在Web开发的早期&#xff0c;前端请求库的唯一选择是使用XMLHttpRequest&#xff08;XHR&#xff09;对象。XHR提供了一种在浏览器中发起HTTP请求的方式&#xff0c;但使用XHR需要编写大量重复的代码&#xff0c;并且缺乏…

网络安全 | 入侵检测系统(IDS)与入侵防御系统(IPS):如何识别并阻止威胁

网络安全 | 入侵检测系统&#xff08;IDS&#xff09;与入侵防御系统&#xff08;IPS&#xff09;&#xff1a;如何识别并阻止威胁 一、前言二、入侵检测系统&#xff08;IDS&#xff09;2.1 IDS 的工作原理2.2 IDS 的技术类型2.3 IDS 的部署方式 三、入侵防御系统&#xff08;…

idea修改模块名导致程序编译出错

本文简单描述分别用Idea菜单、pom.xml文件管理项目模块module 踩过的坑&#xff1a; 通过idea菜单创建模块&#xff0c;并用idea菜单修改模块名&#xff0c;结构程序编译报错&#xff0c;出错的代码莫名奇妙。双击maven弹窗clean时&#xff0c;还是报错。因为模块是新建的&am…

【2024年终总结】深圳工作生活评测

距离上次写年终总结已经过了一年半了&#xff0c;这一年半中哪怕经历了很多的事情&#xff0c;但是感觉又没发生什么。想写一些骚话&#xff0c;却总觉得自己无法完全表达&#xff0c;便也就这样&#xff0c;静静地记录下这一段时光。 现在是2025年&#xff0c;春节前的时光&am…

如何设计浪漫风格的壁纸

一、选择浪漫的色彩 柔和色调&#xff1a; 粉色系&#xff1a;粉色是浪漫的经典色彩&#xff0c;包括淡粉色、玫瑰粉、樱花粉等&#xff0c;能够营造出温馨和甜蜜的氛围。 紫色系&#xff1a;紫色带有神秘和高贵的感觉&#xff0c;如薰衣草紫、淡紫色等&#xff0c;适合营造浪…

element el-table合并单元格

合并 表格el-table添加方法:span-method"” <el-table v-loading"listLoading" :data"SHlist" ref"tableList" element-loading-text"Loading" border fit highlight-current-row :header-cell-style"headClass" …

【unity游戏开发之InputSystem——07】InputSystem+UGUI配合使用(基于unity6开发介绍)

文章目录 一、InputSystem+UGUI配合使用1、官方文档参考2、切换到新的输入模块二、UGUI中的新输入系统输入模块参数相关1、Send Pointer Hover To Parent2、Move Repeat Delay3、Move Repeat Rate4、XR Tracking Origin5、Deselect On Background CLick6、Pointer Behavior7、S…

web端ActiveMq测试工具

如何用vue3创建简单的web端ActiveMq测试工具&#xff1f; 1、复用vue3模板框架 创建main.js,引入APP文件&#xff0c;createApp创建文件&#xff0c;并加载element插件&#xff0c;然后挂载dom节点 2、配置vue.config.js脚本配置 mport { defineConfig } from "vite&qu…

2K高刷电竞显示器推荐

2K高刷电竞显示器推荐&#xff0c;各位喜欢打游戏&#xff0c;身为电竞迷的小伙伴&#xff0c;如果你想选一款2K高刷电竞显示器&#xff0c;那么下面的内容不容错过。 1.HKC G27H4Pro - 2K高刷电竞显示器推荐 外观 - HKC G27H4Pro 2K高刷电竞显示器 初见 HKC G27H4Pro&#x…

他把智能科技引入现代农业领域

江苏田倍丰农业科技有限公司&#xff08;以下简称“田倍丰”&#xff09;是一家专注于粮油种植的农业科技公司&#xff0c;为拥有300亩以上田地的大户提供全面的解决方案。田倍丰通过与当地政府合作&#xff0c;将土地承包给大户&#xff0c;并提供农资和技术&#xff0c;实现利…

第38周:猫狗识别 (Tensorflow实战第八周)

目录 前言 一、前期工作 1.1 设置GPU 1.2 导入数据 输出 二、数据预处理 2.1 加载数据 2.2 再次检查数据 2.3 配置数据集 2.4 可视化数据 三、构建VGG-16网络 3.1 VGG-16网络介绍 3.2 搭建VGG-16模型 四、编译 五、训练模型 六、模型评估 七、预测 总结 前言…

OpenCV2D 特征框架 (6)特征检测与描述类cv::KAZE的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 cv::KAZE 类是 OpenCV 库中用于实现 KAZE 特征检测和描述的类。KAZE 是一种尺度不变特征变换&#xff08;Scale-Invariant Feature Transform, S…

Unity git版本管理

创建仓库的时候添加了Unity的.gitignore模版&#xff0c;在这个时候就能自动过滤不需要的文件 打开git bash之后&#xff0c;步骤git版本管理-CSDN博客 如果报错&#xff0c;尝试重新进git 第一次传会耗时较长&#xff0c;之后的更新就很快了

Neural networks 神经网络

发展时间线 基础概念 多层神经网络结构 神经网络中一个网络层的数学表达 TensorFlow实践 创建网络层 神经网络的创建、训练与推理 推理 推理可以理解为执行一次前向传播 前向传播 前向传播直观数学表达 前向传播直观数学表达的Python实现 前向传播向量化实现 相关数学知识…

2023年吉林省职业院校技能大赛网络系统管理样题

目录 任务清单 &#xff08;一&#xff09;基础配置 &#xff08;二&#xff09;有线网络配置 &#xff08;三&#xff09;无线网络配置 &#xff08;四&#xff09;出口网络配置 附录1&#xff1a;拓扑图​编辑 附录2&#xff1a;地址规划表 任务清单 &#xff08;一&a…

C++入门14——set与map的使用

在本专栏的往期文章中&#xff0c;我们已经学习了STL的部分容器&#xff0c;如vector、list、stack、queue等&#xff0c;这些容器统称为序列式容器&#xff0c;因为其底层是线性序列的数据结构&#xff0c;里面存储的是元素本身。而本篇文章我们要来认识一下关联式容器。 &am…