一个基础又很重要的知识点:JDBC原理(基本案例和面试知识点)
JDBC全称又叫做Java DataBase Connectivity,就是Java数据库连接,说白了就是用Java语言来操作数据库。这篇文章主要是对JDBC的原理进行讲解。不会专注于其使用。主要是理解其思想并对一些常见的面试题进行讲解。
一、JDBC原理
既然JDBC主要是用于java连接数据库的,能连接什么数据库没有指定,其实能连接很多种数据库,而且一般来说可以连接oracle和mysql,通常也是这两种。但是既然JDBC能连接这么多的数据库,开发起来太麻烦了,于是sun公司那帮人想出了一个办法,我定义一套规则,大家都按照这个规则来,实现自己公司访问数据库的实现。这套规则就是JDBC,遵循了JDBC规范的,可以访问自己数据库的API被称之为驱动。
所以jdbc是不变的,但是驱动却有很多种。
现在大家应该能理解了这套原理了。其实是很简单。JDBC只是一套规范接口,真正实现的是下面的各种驱动。我们使用一个例子来解释一下。
二、例子演示
在上面那张图里面,基本上也交代出了使用一个数据库的一般步骤。
(1)注册一个驱动
(2)使用驱动和数据库连接
(3)使用连接对象获取操作数据库的执行对象
我们干脆使用代码来实际演示一遍,首先我在数据库里面建了一张person表,并随便插入了两条记录。
public class JDBCTest01 {
//数据库地址
private static final String url = "jdbc:mysql://localhost:3306/uav";
private static final String name = "com.mysql.jdbc.Driver";
private static final String username = "root";
private static final String password = "root";
private static Connection connection = null;
private static PreparedStatement preparedStatement = null;
private JDBCTest01(String sql){
try{
//步骤一:注册一个驱动
Class.forName(name);
//步骤二:使用驱动和数据库连接
connection = DriverManager.getConnection(url, username, password);
//步骤三:使用连接对象获取操作数据库的执行对象
preparedStatement = connection.prepareStatement(sql);
}catch(Exception e){
e.printStackTrace();
}
}
}
有了这三步我们就能根据返回的preparedStatement操作数据库了,下面我们在main方法中测试一波。
public static void main(String[] args) {
String sql = "SELECT * FROM person";
JDBCTest01 dbManager = new JDBCTest01(sql);
String id, name, age;
try{
ResultSet result = dbManager.preparedStatement.executeQuery();
while(result.next()){
id = result.getString(1);
name = result.getString(2);
age = result.getString(3);
//显示出每一行数据
System.out.println(id + " " + name + " " + age);
}
result.close();
dbManager.close();
connection.close();
preparedStatement.close();
}catch (Exception e){
e.printStackTrace();
}
}
用完了之后关闭就好。到现在为止一个完整的JDBC案例也给出来了。现在我们来分析一下使用一个数据库的一般步骤。针对以上3各功能,提供了一下4个类:
(1)DriverManager:该类管理数据库驱动程序。
(2)Connection:管理数据库建立的连接。
(3)Statement:负责将要执行的sql体局提交到数据库。
(4)ResultSet:执行sql查询语句返回的结果集。
1、注册驱动
JDBC中规定,驱动类在被加载时,需要自己“主动”把自己注册到DriverManger中,如何注册一个驱动呢?上面我们好像使用的是反射,但是反射只是实现了注册的功能,追究其原理,我们还需要到com.mysql.jdbc.Driver类的源代码中找寻答案。
public class Driver extends NonRegisteringDriver implements java.sql.Driver {
static {
try {
java.sql.DriverManager.registerDriver(new Driver());
} catch (SQLException E) {
throw new RuntimeException("Can't register driver!");
}
}
}
也就是说我们注册驱动的时候只是new了自己,也就是Driver,既然这样我们直接把注册驱动类的代码修改为加载驱动类。也可以实现同样的功能,于是就使用Class.forName(“com.mysql.jdbc.Driver”); 代替了,形式也更加的简单。
2、获取连接
在分析原理的时候意思就是有了驱动,我们还要和我们的数据库建立连接。这个很简单,既然是数据库,我们首先需要指定我们使用的数据库是哪一个,还有用户名和密码。
3、获取Statement
获取了连接之后,下面我们就可以获取Statement。Statement是用来向数据库发送要执行的SQL语句的。
Statement最为重要的方法是:
(1)int executeUpdate(String sql):执行更新操作,即执行insert、update、delete语句,其实这个方法也可以执行create table、alter table,以及drop table等语句,但我们很少会使用JDBC来执行这些语句;
(2)ResultSet executeQuery(String sql):执行查询操作,执行查询操作会返回ResultSet,即结果集。
4、读取结果集中的数据
我们在main方法中测试了一下,首先定义了一条sql语句,然后使用Statement向数据库发送我们的sql语句。此时会返回一个结果集ResultSet。
ResultSet就是一张二维的表格,我们可以调用rs对象的next()方法把“行光标”向下移动一行,当第一次调用next()方法时,“行光标”就到了第一行记录的位置,这时就可以使用ResultSet提供的getXXX(int col)方法来获取指定列的数据了。当然里面的方法还很多。
(1)String getString(int columnIndex):获取指定列的String类型数据;
(2)int getInt(int columnIndex):获取指定列的int类型数据;
(3) double getDouble(int columnIndex):获取指定列的double类型数据;
(4)boolean getBoolean(int columnIndex):获取指定列的boolean类型数据;
(5)Object getObject(int columnIndex):获取指定列的Object类型的数据。
(6)String getString(String columnName):获取名称为columnName的列的String数据;
(7)int getInt(String columnName):获取名称为columnName的列的int数据;
(8)double getDouble(String columnName):获取名称为columnName的列的double数据;
(9)boolean getBoolean(String columnName):获取名称为columnName的列的boolean数据;
(10)Object getObject(String columnName):获取名称为columnName的列的Object数据;
上面其实分了两类。一类是根据指定列,一类是根据指定列名。
当然如果执行失败了呢?是否要支持滚动呢?这要从Connection类的createStatement()方法说起。也就是说创建的Statement决定了使用Statement创建的ResultSet是否支持滚动。
Statement createStatement(int resultSetType, int resultSetConcurrency)
resultSetType的可选值:
(1)ResultSet.TYPE_FORWARD_ONLY:不滚动结果集;
(2)ResultSet.TYPE_SCROLL_INSENSITIVE:滚动结果集,但结果集数据不会再跟随数据库而变化;
(3)ResultSet.TYPE_SCROLL_SENSITIVE:滚动结果集,但结果集数据不会再跟随数据库而变化;
可以看出,如果想使用滚动的结果集,我们应该选择TYPE_SCROLL_INSENSITIVE!其实很少有数据库驱动会支持TYPE_SCROLL_SENSITIVE的特性!通常我们也不需要查询到的结果集再受到数据库变化的影响。
现在来看基本上JDBC就这么多内容,还有最后一个问题。那就是sql注入攻击的问题。
三、sql注入攻击
为了解释好他的概念,我们使用java关键字来讲解,我们在写代码的时候都知道Class是java中的一个关键字,我们不能把它当成一个普通变量来定义,对于sql也是同样的道理,比如select是一个查询关键字,我们就不能把它当成一个普通的字段来操作。
但是这样会出现一个问题,我们的用户总是千奇百怪,你不知道他会做出什么样的事,万一用户传入的数据中包含sql关键字时,就有可能通过这些关键字改变sql语句的语义,从而执行一些特殊的操作,这样的攻击方式就叫做sql注入攻击。
JDBC是如何解决这个问题的呢?还要从我们的第三步获取Statement说起。PreparedStatement利用预编译的机制将sql语句的主干和参数分别传输给数据库服务器,从而使数据库分辨的出哪些是sql语句的主干哪些是参数,这样一来即使参数中带了sql的关键字,数据库服务器也仅仅将他当作参数值使用,从而从原理上防止了sql注入的问题。
一句话来总结就是把主干和参数分别传输。
其实不仅仅有PreparedStatement,还有一个Statment类也能执行sql语句,功能类似。但是稍微有一点区别,PreparedStatement 与Statment比较:
语法不同:PreparedStatement可以使用预编译的sql,而Statment只能使用静态的sql
效率不同: PreparedStatement可以使用sql缓存区,效率比Statment高
安全性不同: PreparedStatement可以有效防止sql注入,而Statment不能防止sql注入。
一个基础又很重要的知识点:JDBC原理(基本案例和面试知识点)的更多相关文章
- Java 面试知识点解析(五)——网络协议篇
前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...
- Java 面试知识点解析(六)——数据库篇
前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...
- Java 面试知识点解析(七)——Web篇
前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...
- 你要的 React 面试知识点,都在这了
摘要: 问题很详细,插图很好看. 原文:你要的 React 面试知识点,都在这了 作者:前端小智 Fundebug经授权转载,版权归原作者所有. React是流行的javascript框架之一,在20 ...
- Java面试知识点汇总
Java面试知识点汇总 置顶 2019年05月07日 15:36:18 温柔的谢世杰 阅读数 21623 文章标签: 面经java 更多 分类专栏: java 面试 Java面试知识汇总 版权声明 ...
- 1 开发一个注重性能的JDBC应用程序不是一件容易的事. 当你的代码运行很慢的时候JDBC驱动程序并不会抛出异常告诉你。 本系列的性能提示将为改善JDBC应用程序的性能介绍一些基本的指导原则,这其中的原则已经被许多现有的JDBC应用程序编译运行并验证过。 这些指导原则包括: 正确的使用数据库MetaData方法 只获取需要的数据 选用最佳性能的功能 管理连
1 开发一个注重性能的JDBC应用程序不是一件容易的事. 当你的代码运行很慢的时候JDBC驱动程序并不会抛出异常告诉你. 本系列的性能提示将为改善JDBC应用程序的性能介绍一些基本的指导原则,这其中的 ...
- Java---常用基础面试知识点
综合网上的一点资源,给大家整理了一些Java常用的基础面试知识点,希望能帮助到刚开始学习或正在学习的学员. 1.抽象 抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方 ...
- Java 面试知识点解析(一)——基础知识篇
前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...
- 用一个下午从零开始搭建一个基础lbs查询服务
背景 现在做一个sns如果没有附近的功能,那就是残缺的.网上也有很多现成的lbs服务,封装的很完整了. 我首先用了下百度lbs云,但是有点不适合自己的需要,因此考虑用mongodb建一个简单的lbs服 ...
随机推荐
- python+树莓派实现IoT(物联网)数据上传到服务器
环境:raspbian-stretch(2018-06-27) 树莓派:3代B型 1.树莓派设备,需要在野外也能拥有独立联网能力,那必不可少的需要使用物联网模块. 这里使用的是微雪的SIM868通讯模 ...
- 树莓派3B 安装gcc和g++
转:https://blog.csdn.net/zhuming3834/article/details/81946707 安装 如果不是root 用户,请自行加上sudo apt-get instal ...
- 【C语言】判断某一正整数是否为完数
什么是完数? 如果一个数等于它的因子之和,则称该数为“完数”(或“完全数”). 例如,6的因子为1.2.3,而 6=1+2+3,因此6是“完数”. 程序框图:m 问题分析 根据完数的定义,解决本题的 ...
- 【MySQL】单表查询
" 目录 where 约束 group by 分组查询 聚合函数 having 过滤 order by 查询排序 limit 限制查询的记录数 # 语法 select 字段1, 字段2 .. ...
- iOS 实现 摇一摇
摇一摇功能:调用了系统自带加速器,当设备摇动时,系统会 计算出加速器的加速值,然后告诉设备是否发生摇动手势,系统只会运动开始和结束时通知你,并不会在运动发生的整个过程中始终向你报告每一次运动.例如,你 ...
- 安卓之文本视图TextView及跑马灯效果
一.基本属性和设置方法 二.跑马灯用到的属性与方法说明 三.省略方式的取值说明 四.跑马灯效果案例代码 (1)布局xml文件 <?xml version="1.0" en ...
- Codeforces Round #576 (Div. 2) 题解
比赛链接:https://codeforc.es/contest/1199 A. City Day 题意:给出一个数列,和俩个整数\(x,y\),要求找到序号最靠前的数字\(d\),使得\(d\)满足 ...
- QSS基础-设置控件样式
1.QSS基础-设置同类控件样式'''QSS基础:Qt Style SheetQt样式表用于设置控件的样式和风格(比如控件的背景色,字体颜色,字体大小等)和CSS的功能比较相似,功能相似''' fro ...
- 一个支持国密SM2/SM3/SM4/SM9/ZUC/SSL的密码工具箱
转:https://blog.csdn.net/xuq09/article/details/91815366 The GmSSL Project网址:http://gmssl.org/docs/qui ...
- java中对于多态的一个实例分析
首先来看这样的一段代码,其中对于类的定义如下: class Parent{ public int myValue=100; public void printValue() { System.out. ...