JDBC(Java DataBase Connectivity )

图片来源:动力节点老杜的JDBC视频讲解

JDBC(Java DataBase Connectivity )

  • 一、JDBC 的本质
  • 二、开始前的准备工作
  • 三、关于 JDBC 中的事务
  • 四、JDBC 编程六步
    • 1.注册驱动
    • 2.获取连接
    • 3.获取数据库操作对象
    • 4.执行SQL语句
    • 5.处理结果查询集
    • 6.释放资源
  • 五、JDBC工具类
  • 六、DataSource


一、JDBC 的本质

  • JDBC(Java DataBase Connectivity),即Java连接数据库。也就是通过Java语言来操作数据库。
  • JDBC就是官方定义的一套接口。在java.sql.*包下。
  • 数据库厂家实现了这一套接口,对于我们程序员而言,就是调用这个接口中的方法就可以完成对数据库的操作。
  • 数据库的jar包就是JDBC这一套接口的实现类而已。
    图片来源——动力节点老杜的JDBC视频

二、开始前的准备工作

  • 首先从官网中下载对应数据库的jar包。
  • 在CLASSPATH中配置数据库的驱动(就是一个jar包)。===> 导入到自己的项目中。

三、关于 JDBC 中的事务

  • 默认为自动提交。
    conn.setAutoCommit(false);//关闭自动提交机制
    conn.rollback();//出错了回滚
    conn.commit();//提交事务
    

四、JDBC 编程六步

  1. 注册驱动(作用:告诉Java程序,即将连接的是哪个品牌的数据库)
  2. 获取连接(表示 JVM 的进程和数据库进程之间的通道打开了,这属于进程之间的通信,重量级的,使用完之后一定要关闭)
  3. 获取数据库操作对象(专门执行sql语句的对象)
  4. 执行SQL语句(DQL、DDL、DML…)
  5. 处理查询结果集(只有当第四步执行的是select语句的时候,才有第五步处理查询结果集)
  6. 释放资源(使用完资源以后一定要关闭资源。Java和数据库属于进程间的通信,开启后一定要关闭)

1.注册驱动

  • 第一种方式

    //DriverManager.registerDriver(new com.mysql.cj.jdbc.Driver());
    Driver driver = new com.mysql.cj.jdbc.Driver();//多态,父类型引用指向子类型对象
    DriverManager.registerDriver(driver);
    
  • 第二种方式

    # properties 文件,文件名为 jdbc.properties
    driver=com.mysql.cj.jdbc.Driver
    
    //1.注册驱动器
    //以下方式不需要接收返回值,因为我们只想用他类加载动作。
    //静态代码块,在类加载是执行(因为数据库实现的Driver类的静态代码块中写了第一种方式的代码)
    //Class.forName("com.mysql.cj.jdbc.Driver");
    //因为参数是一个字符串,字符串可以写到xxx.properties文件中
    ResourceBundle bundle = ResourceBundle.getBundle("jdbc");
    String driver = bundle.getString("driver");
    Class.forName(diver);
    
  • 对于 ResourceBundle和Properties 文件的使用参考我都博客 ====> properties 属性配置文件

  • 第二种方式引入Java中的properties文件可以提高程序的扩展性。

2.获取连接

  • 第一种方式

    /*url是什么?
    	* 统一资源定位符(网络中某个资源的绝对路径)
    	* jdbc:mysql://localhost:3306/gdb
    	* jdbc:mysql://——协议
    	* 通信协议是通信之前就提前定好的数据传送格式,数据包具体怎么传输据,格式提前定好的。
    	* localhost IP地址
    	* 3306 mysql数据库端口号
    	* gdb 具体的数据库实例名
    	* 说明localhost和127.0.0.1都是本机IP地址
    */
    String url = "jdbc:mysql://127.0.0.1:3306/jdbc"; 	//"jdbc:mysql://ip:port/数据库名字"
    String user = "root";
    String password = "123456";
    Connection coon = DriverManager.getConnection(url, user, password);
    
  • 第二种方式

    # properties 文件,文件名为 jdbc.properties
    driver=com.mysql.cj.jdbc.Driver
    url=jdbc:mysql://localhost:3306/gdb
    user=root
    password=123456
    
    //使用资源绑定器绑定属性配置文件
    ResourceBundle bundle = ResourceBundle.getBundle("jdbc");
    String url = bundle.getString("url");
    String user = bundle.getString("user");
    String password = bundle.getString("password");
    Connection coon = DriverManager.getConnection(url, user, password);
    
  • 使用第二种方式的目的也是为了提高程序的扩展性、降低耦合度。

3.获取数据库操作对象

  • 第一种方式

    Statement stmt = coon.createStatement();
    
  • 第二种方式

    //上面的方式存在sql注入的问题
    /**
    	* 1.解决sql注入的问题 只要用户的信息不参与sql语句的编译过程,问题就解决了。
    	* 即使用户提供的信息中含有SQL语句的关键字,但是没有参与编译,不起作用。 必须使用java.sql.PreparedStatement
    	* PreparedStatement接口继承了java.sql.Statement
    	* PreparedStatement接口是属于预编译的数据库操作对象。
    	* PreparedStatement的原理是,预先对SQL语句的框架进行编译,然后再给SQL语句传”值“。
    */
    // 3.获取预编译的数据库操作对象
    /*SQl语句的框子
    	* 一个?表示一个站位符,一个?将来接收一个”值“,注意占位符不能用''括起来
     */
    String sql = "select * from t_user where loginName = ? and loginPassword = ?";
    //程序执行到这里,会发送sql语句框架,然后DBMS进行sql语句的预先编译
    PreparedStatement ps = conn.prepareStatement(sql);
    //第一个?下标为1,第二个?下标为2
    ps.setString(1,userLoginInfo.get("userName"));
    ps.setString(2,userLoginInfo.get("password"));
    
  • Statement和PreparedStatement对比:

    • Statement存在sql注入的问题,PreparedStatement解决了sql注入问题。
    • Statement是编译一次执行一次。PreparedStatement是编译一次可以执行N次。PreparedStatement的效率高一些。
    • PreparedStatement会在编译阶段做类型安全检查。
  • 什么情况必须使用Statement?

    • 要求进行sql语句中的关键字拼写到sql语句中时必须使用Statement数据库操作对象。

4.执行SQL语句

  • 第一种方式:对应了获取数据库操作对象的方式一

    //4.执行sql语句
    //JDBC的sql语句不需要提供分号结尾
    //String sql = "delete from dept where deptno=50";
    String sql = "insert into dept values (50,'人事部','成都')";
    
    //专门执行DML语句的(insert delete update)
    //返回值是”影响数据库中的记录条数“
    int count = stmt.executeUpdate(sql);
    
    //专门执行DQL语句的方法(select)
    String sql = "select empno as a,ename,sal from emp";
    ResultSet rs = stmt.executeQuery(sql);
    
  • 第二种方式:对应了获取数据库操作对象的方式二

    ResultSet rs = ps.executeQuery();//注意这里不能再传sql语句了
    

5.处理结果查询集

// 5.处理查询结果集
while(rs.next()) {
//int empno = rs.getInt("a");//列名称不是查询表中的名称,是查询结果集的名称
//String ename = rs.getString("ename");
//double sal = rs.getDouble("sal");
    int empno = rs.getInt(1);
	String ename = rs.getString(2);
	double sal = rs.getDouble(3);
	System.out.println(empno + "   " + ename + "   " + sal);
}

6.释放资源

finally {
    // 6.释放资源
    if (rs != null)
        try {
			rs.close();
        } catch (Exception e) {
				e.printStackTrace();
			}
    if (stmt != null)
		try {
			stmt.close();
		} catch (Exception e) {
			e.printStackTrace();
	}
	if (conn != null)
		try {
			conn.close();
		} catch (Exception e) {
			e.printStackTrace();
		}
}

五、JDBC工具类

  • 编写一个 JDBC 工具类,可以简化 JDBC 的开发过程。
/**
 * JDBC工具类,简化JDBC编程
 */
public class DBUtil {
    public static ResourceBundle bundle = ResourceBundle.getBundle("jdbc/resources/jdbc");
    public static String driver = bundle.getString("driver");
    public static String url = bundle.getString("url");
    public static String user = bundle.getString("user");
    public static String password = bundle.getString("password");
    
    //静态代码快在类加载的时候执行,只执行一次
    static {
        try {
            Class.forName(driver);//只让类加载
        } catch (ClassNotFoundException e) {
            System.out.println("类加载失败");
        }
    }

    /**
     * 工具类中的构造方法都是私有的
     * 因为工具类当中的方法都是静态的,不需要new对象,直接采用类名.调用
     */
    private DBUtil(){}

    /**
     * 获取数据库连接对象
     * @return 连接对象
     */
    public static Connection getConnection() throws SQLException{
        return DriverManager.getConnection(url,user,password);
    }

    /**
     * 释放资源
     * @param conn 连接对象
     * @param ps    数据库操作对象
     * @param rs    结果集
     */
    public static void close(Connection conn, Statement ps, ResultSet rs){//注意这里Statement这里是想想抽象编程,多态
        if (rs != null){
            try {
                rs.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (ps != null){
            try {
                ps.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (conn != null){
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
    }
    /**
     * 释放资源
     * @param conn 连接对象
     * @param ps    数据库操作对象
     */
    public static void close(Connection conn, PreparedStatement ps){
        if (ps != null){
            try {
                ps.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        if (conn != null){
            try {
                conn.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
    }
}


六、DataSource

  • DataSource 就是数据源。数据源表示数据的来源,从某个 IP 上的数据库能够获取数据。javax.sql.DataSource 接口表示数据源,提供了标准的方法获取与数据库绑定的连接对象(Connection)。(生成Connection对象是数据源负责的)。
  • java.sql.Connection 是连接对象,在 Connection 上能够从程序代码发送查询命令,更新数据的语句给数据库;同时从 Connection 获取命令的执行结果。Connection很重要,像一个电话线把应用程序和数据库连接起来。
    在这里插入图片描述
  • 数据源我们自己也可以编写,实现了 javax.sql.DataSource 接口的就是数据源。
    package com.gdb.jdbc;
    
    import javax.sql.DataSource;
    import java.io.PrintWriter;
    import java.sql.Connection;
    import java.sql.DriverManager;
    import java.sql.SQLException;
    import java.sql.SQLFeatureNotSupportedException;
    import java.util.logging.Logger;
    
    public class MyDataSource implements DataSource {
        // 添加4个属性
        private String driver;
        private String url;
        private String username;
        private String password;
    
        // 提供4个setter方法
        public void setDriver(String driver) {
            this.driver = driver;
        }
    
        public void setUrl(String url) {
            this.url = url;
        }
    
        public void setUsername(String username) {
            this.username = username;
        }
    
        public void setPassword(String password) {
            this.password = password;
        }
    
        // 重点写怎么获取Connection对象就行。其他方法不用管。
        @Override
        public Connection getConnection() throws SQLException {
            try {
                Class.forName(driver);
                Connection conn = DriverManager.getConnection(url, username, password);
                return conn;
            } catch (Exception e) {
                e.printStackTrace();
            }
            return null;
        }
    
        @Override
        public Connection getConnection(String username, String password) throws SQLException {
            return null;
        }
    
        @Override
        public PrintWriter getLogWriter() throws SQLException {
            return null;
        }
    
        @Override
        public void setLogWriter(PrintWriter out) throws SQLException {
    
        }
    
        @Override
        public void setLoginTimeout(int seconds) throws SQLException {
    
        }
    
        @Override
        public int getLoginTimeout() throws SQLException {
            return 0;
        }
    
        @Override
        public Logger getParentLogger() throws SQLFeatureNotSupportedException {
            return null;
        }
    
        @Override
        public <T> T unwrap(Class<T> iface) throws SQLException {
            return null;
        }
    
        @Override
        public boolean isWrapperFor(Class<?> iface) throws SQLException {
            return false;
        }
    }
    

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

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

相关文章

SpringBoot_基础

学习目标 基于SpringBoot框架的程序开发步骤 熟练使用SpringBoot配置信息修改服务器配置 基于SpringBoot的完成SSM整合项目开发 一、SpringBoot简介 1. 入门案例 问题导入 SpringMVC的HelloWord程序大家还记得吗&#xff1f; SpringBoot是由Pivotal团队提供的全新框架&…

docker指令存档

目录 Docker 1、概念 2、架构图 3、安装 4、Docker怎么工作的&#xff1f; 5、Docker常用命令 帮助命令 镜像命令 1、查看镜像 2、帮助命令 3、搜索镜像 4、拉取镜像 5、删除镜像 容器命令 1、启动 2、查看运行的容器 3、删除容器 4、启动&停止 其他命令…

苹果Find My市场需求火爆,伦茨科技ST17H6x芯片助力客户量产

苹果发布AirTag发布以来&#xff0c;大家都更加注重物品的防丢&#xff0c;苹果的 Find My 就可以查找 iPhone、Mac、AirPods、Apple Watch&#xff0c;如今的Find My已经不单单可以查找苹果的设备&#xff0c;随着第三方设备的加入&#xff0c;将丰富Find My Network的版图。产…

蓝桥杯备战——3.定时器前后台

1.STC15F2k61S2的定时器 阅读STC15系列的手册&#xff0c;我们可以看到跟STC89C52RC的定时器还是有不同之处的&#xff1a; 由上图可以看到我们可以通过AUXR寄存器直接设置定时器的1T/12T模式了 在定时器0/1模式上也可以设置为16位自动重装载。 另外需要注意IAP15F2K61S2只有…

python 学习之 re库的基本使用(正则匹配)上

目录 一、基本用法 二、函数介绍 1、match函数 2、search 函数 3、compile 函数 4、findall 和 finditer 函数 5、sub 函数和 subn 函数 6、split 函数 一、基本用法 首先我们需要引入 re 库 代码基本框架使用两行代码实现 测试代码&#xff1a; import reret re.m…

Linux管道学习(无名管道)

目录 1、概述 2、管道的创建 3、管道读写行为 3.1、管道读 在linux中管道有两种&#xff0c;一是无名管道&#xff08;匿名管道&#xff09;&#xff0c;第二种是有名管道&#xff1b;无名管道主要用于有血缘关系的父子进程间通信&#xff0c;有名管道则不受该限制&#xf…

描绘未知:数据缺乏场景的缺陷检测方案

了解更多方案内容&#xff0c;欢迎您访问官网&#xff1a;neuro-T | 友思特 机器视觉 光电检测&#xff1b;或联系销售经理&#xff1a;18124130753 导读&#xff1a; 深度学习模型帮助工业生产实现更加精确的缺陷检测&#xff0c;但其准确性可能受制于数据样本的数量。友思特…

Gradle学习笔记:Gradle的简介、下载与安装

文章目录 一、什么是Gradle二、为什么选择Gradle三、下载并安装Gradle四、Gradle的bin目录添加到环境变量五、测试Gradle是否安装正常 一、什么是Gradle Gradle是一个开源构建自动化工具&#xff0c;专为大型项目设计。它基于DSL&#xff08;领域特定语言&#xff09;编写&…

科大讯飞 再次引爆Ai

去年「科大讯飞版ChatGPT」星火大模型刚上线的时候&#xff0c;小编给大家推荐过一波&#xff0c;演示了其强大的功能&#xff0c;不少小伙伴都立马申请体验了一把&#xff0c;有小伙伴还私信我说功能非常强大&#xff0c;工作效率提高不少&#xff0c;支持国产大模型之类赞扬。…

idea 打包跳过测试

IDEA操作 点击蓝色的小球 手动命令 mvn clean package -Dmaven.test.skiptrue

音频格式之AAC:(2)AAC封装格式ADIF,ADTS,LATM,extradata及AAC ES存储格式

系列文章目录 音频格式的介绍文章系列&#xff1a; 音频编解码格式介绍(1) ADPCM&#xff1a;adpcm编解码原理及其代码实现 音频编解码格式介绍(2) MP3 &#xff1a;音频格式之MP3&#xff1a;(1)MP3封装格式简介 音频编解码格式介绍(2) MP3 &#xff1a;音频格式之MP3&#x…

unity学习笔记----游戏练习07

一、僵尸攻击和植物的掉血和销毁 当僵尸接触到植物开始攻击时会持续削减植物的血量&#xff0c;当植物血量为零时就销毁当前植物。 在plantManager中&#xff0c; 为植物添加一个血量HP 100&#xff0c; public int HP 100; 在写一个减少血量的方法&#xff0c;来减少血…

【RabbitMQ】交换机的概念及使用

一、引言 1、什么是交换机 RabbitMQ中&#xff0c;交换机是一个核心概念&#xff0c;主要用来将生产者生产出来的消息&#xff0c;传送到对应的队列中。实际上&#xff0c;生产者生产的消息从不会直接发送到队列&#xff0c;而是发送到交换机。交换机一方面接收来自生产者的消…

golang入门

学习方法 1、在实践中学 2、适当的囫囵吞枣&#xff0c;有可能学到后面&#xff0c;对前面的疑问焕然大悟 3、注重整体&#xff0c;刚开始不要去扣细节 安装 需要配置3个环境变量&#xff0c;如果.msi文件安装时设置好了就不需要了&#xff0c;自己可以检查下 GOROOT&…

【Unity】粒子贴图异常白边问题

从PS制作的黑底&#xff0c;白光的贴图。放入Unity粒子中&#xff0c;拉远看会有很严重的白边&#xff0c;像马赛克一样。 材质使用&#xff1a;Mobile/Particles/Additive 经测试只使用一张黑色的图片&#xff0c;也会有白边。 解决方案&#xff1a; 关闭黑色底&#xf…

web前端之不一样的居中方式、解决tabBar选项卡居中问题、css支持嵌套、auto

MENU 前言htmlstyle效果 前言 这里不能使用justify-content: center;&#xff0c;因为在小屏幕上&#xff0c;这种方式无法显示最前面的两个tabBar。 html <div id"box" class"d_f o_a mt_50 mb_50 ml_20 mr_20"><div class"ws_n">…

【网站项目】新冠疫情隔离人员信息管理系统(有源码)

🙊作者简介:多年一线开发工作经验,分享技术代码帮助学生学习,独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。🌹赠送计算机毕业设计600个选题excel文件,帮助大学选题。赠送开题报告模板,帮助书写开题报告。作者完整代码目录供你选择: 《Springboot网站项目…

Hadoop3.x学习笔记

文章目录 一、Hadoop入门1、Hadoop概述1.1 简介1.2 hadoop优势1.3 hadoop组成1.4 大数据技术生态体系 2、环境准备(重点)2.1 模板机配置2.2 模板创建 3、本地运行模式&#xff08;官方WordCount&#xff09;4、Hadoop集群搭建(&#x1f31f;重点)4.1 环境准备(集群分发脚本xsyn…

[GXYCTF2019]BabyUpload1

尝试各种文件&#xff0c;黑名单过滤后缀ph&#xff0c;content-type限制image/jpeg 内容过滤<?&#xff0c;木马改用<script languagephp>eval($_POST[cmdjs]);</script> 上传.htaccess将上传的文件当作php解析 蚁剑连接得到flag

水波浪标题

上图效果要先复制第13次修改的备忘录&#xff0c;再另外保存下面的代码&#xff1a; <!DOCTYPE html> <html lang"zh"> <a class"a-href a-h">水波浪标题</a> <style>.h1-div {/* 隐藏 */display: none;}h1 {display: inli…