JavaWeb之JDBC
一、介绍
C#定义了ADO.Net接口来实现对SQLServer、Oracel等数据库的访问,那Java定义了JDBC接口实现对数据库的访问,数据库提供商只要实现这些接口,Java语言就能访问数据库。
二、工作流程
1、驱动引入
数据库提供商实现了JDBC接口,可好几种数据库,Java知道是哪一种呢,于是乎得注册不同的驱动来操作对应的数据库,注册驱动也得要有驱动才是,所以首先是将驱动引入项目,这里用Mysql举例,之前也用C#操作过Mysql数据库,今天用Java操作,驱动下载地址:https://dev.mysql.com/downloads/file/?id=468319,将下载的文件解压,会找到mysql-connector-java-5.1.41-bin.jar文件,把该文件放入web-Info/lib下,这样就完成了驱动的引入。
2、注册驱动
引入驱动之后Java也不知道是用的什么数据库,所以得注册一下才知道是谁,注册之后会返回对应的驱动管理对象,就和入职一样,你到公司了但不报到那也不知道来了没来,报到了才会有针对个人的流程。
3、创建连接
数据库可能存放在远程,那怎么和数据库搭上呢,这就需要连接。
4、执行操作
连接上之后要干嘛呢,不能一直连着不干事情啊,所以连接之后执行数据库的操作,增删改查等。
5、返回结果
增删改查完了,总要有个回信吧,不然怎么知道成功与否,查询的话会返回查询的数据,增加、删除、修改会返回影响的行数。
6、释放
把结果也返回了,但不能老连着数据库啊,这样也占用资源,创建的对象也没释放,还占空间,所以用完了就把它关掉。
上面大致把操作数据库的流程列了下,下面通过实验来操作一下。
7、验证
这里先在本地数据库创建一个数据库testdb,然后在表中创建一个users表,有id、name、age、birthday字段,代表着不同的数据类型。

<%@page import="com.mysql.jdbc.Driver"%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ page import="java.sql.*" %>
<%@ page import="java.util.*" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<% Connection conn=null;
Statement stmt=null;
ResultSet rs=null;
try{
//注册驱动
DriverManager.registerDriver(new com.mysql.jdbc.Driver());
//通过注册的驱动获得连接对象Connection
conn=DriverManager.getConnection("jdbc:mysql://localhost:3306/testdb","root","123456");
//通过Statement对象执行操作 返回结果ResultSet
stmt=conn.createStatement();
//返回结果
rs=stmt.executeQuery("select * from users"); while(rs.next())
{
out.println("姓名:"+rs.getString("name")+" 年龄:"+rs.getInt("age") +" 出生日期:"+rs.getDate("birthday"));
}
}
catch(SQLException e)
{
out.println(e.getMessage());
e.printStackTrace();
}
finally
{
//释放资源
if(conn!=null)
{
conn.close();
}
if(stmt!=null)
{
stmt.close();
}
if(rs!=null)
{
rs.close();
}
}
%>
</body>
</html>

二、带参数的sql执行
上面使用Statement来查询数据库,但是Statement不能使用带参数的sql,如果不带参数直接拼接sql语句则会有注入式攻击的风险,那如果要使用带参数的sql就需要用它的子类对象:PreparedStatement.用它可以防止注入式攻击。
<%@page import="com.mysql.jdbc.Driver"%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ page import="java.sql.*" %>
<%@ page import="java.util.*" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<% Connection conn=null; ResultSet rs=null;
PreparedStatement prestmt=null;
try{
//注册驱动
DriverManager.registerDriver(new com.mysql.jdbc.Driver());
//通过注册的驱动获得连接对象Connection
conn=DriverManager.getConnection("jdbc:mysql://localhost:3306/testdb","root","123456");
String sql="insert into users (name,age,birthday) values(?,?,?)"; prestmt=conn.prepareStatement(sql,Statement.RETURN_GENERATED_KEYS);
prestmt.setString(1, "cuiyw");
prestmt.setInt(2, 25);
prestmt.setDate(3,new java.sql.Date(new java.util.Date().getTime()));
int result=prestmt.executeUpdate(); if(result>0)
{
out.println("新增成功");
rs=prestmt.getGeneratedKeys();
while(rs.next())
{
out.println("生成的主键ID为:"+rs.getInt(1));
}
}
}
catch(SQLException e)
{
out.println(e.getMessage());
e.printStackTrace();
}
finally
{
//释放资源
if(conn!=null)
{
conn.close();
}
if(prestmt!=null)
{
prestmt.close();
}
if(rs!=null)
{
rs.close();
}
}
%>
</body>
</html>
上面的代码实现了两个功能,一通过往users表中插入一条记录,使用PreparedStatement带参数的sql,二获取自动增长列的ID。ADO.Net中使用@@Identity,jdbc中使用getGeneratedKeys(),不过在实例化PreparedStatement的时候要带上Statement.RETURN_GENERATED_KEYS。


三、批处理
Statement、PreparedStatement能一次执行新增、修改、删除多条sql语句。一般批处理和事务一起使用,比如转账等。下面是事务批处理插入数据。
<%@page import="com.mysql.jdbc.Driver"%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ page import="java.sql.*" %>
<%@ page import="java.util.*" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<% Connection conn=null; ResultSet rs=null;
PreparedStatement prestmt=null;
try{
//注册驱动
DriverManager.registerDriver(new com.mysql.jdbc.Driver());
//通过注册的驱动获得连接对象Connection
conn=DriverManager.getConnection("jdbc:mysql://localhost:3306/testdb?rewriteBatchedStatements=true","root","123456");
String sql="insert into users (name,age,birthday) values(?,?,?)"; //开启事务
conn.setAutoCommit(false);
prestmt=conn.prepareStatement(sql); for(int i=0;i<2;i++)
{ prestmt.setString(1, "cuiyw"+i);
prestmt.setInt(2, 25+i);
prestmt.setDate(3,new java.sql.Date(new java.util.Date().getTime()));
prestmt.addBatch();
}
//批处理
prestmt.executeBatch();
//提交事务
conn.commit(); conn.setAutoCommit(true); }
catch(SQLException e)
{
//回滚
conn.rollback();
out.println(e.getMessage());
e.printStackTrace();
}
finally
{
//释放资源
if(conn!=null)
{
conn.close();
}
if(prestmt!=null)
{
prestmt.close();
}
if(rs!=null)
{
rs.close();
}
}
%>
</body>
</html>
四、调用存储过程
在使用数据库的过程中,可能会调用存储过程,可以使用CallableStatement来调用存储过程。
调用存储函数 1.{?= call <procedure-name>[(<arg1>,<arg2>, ...)]}
调用存储过程 2.{call <procedure-name>[(<arg1>,<arg2>, ...)]}
通过CallableStatement对象的registerOutParameter() 方法注册Out参数
通过CallableStatement对象的setXxx()方法设定IN或In out 参数,若想将参数设为null,可以使用setNUll()
如果所调用的是带返回参数的存储过程还需要通过CallableStatement对象的getXxx()获取输出参数
这里先创建了一个存储过程在数据库中,有一个输入参数一个输出参数

<%@page import="com.mysql.jdbc.Driver"%>
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ page import="java.sql.*" %>
<%@ page import="java.util.*" %>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<% Connection conn=null; ResultSet rs=null;
CallableStatement callstmt=null;
try{
//注册驱动
DriverManager.registerDriver(new com.mysql.jdbc.Driver());
//通过注册的驱动获得连接对象Connection
conn=DriverManager.getConnection("jdbc:mysql://localhost:3306/testdb?rewriteBatchedStatements=true","root","123456");
callstmt=conn.prepareCall("{call selectUserById(?,?)}");
callstmt.setInt(1, 2);
callstmt.registerOutParameter(2, Types.INTEGER);
rs=callstmt.executeQuery();
while(rs.next())
{
out.println("姓名:"+rs.getString("name")+" 年龄:"+rs.getInt("age") +" 出生日期:"+rs.getDate("birthday"));
}
out.println("存储过程返回值:"+callstmt.getInt(2));
}
catch(SQLException e)
{ out.println(e.getMessage());
e.printStackTrace();
}
finally
{
//释放资源
if(conn!=null)
{
conn.close();
}
if(callstmt!=null)
{
callstmt.close();
}
if(rs!=null)
{
rs.close();
}
}
%>
</body>
</html>

五、其他
对于jdbc操作二进制数据,比如图片,元数据、连接池就不一一介绍了,在开发中如果直接使用jdbc,这些也都会封装成一个类,现在的项目也一般使用框架来映射数据库才操作。
JavaWeb之JDBC的更多相关文章
- JavaWeb用Jdbc操作MySql数据库(一)
一.添加开发包.在JavaWeb中用jdbc操作数据库,使用方法与java一样,但是在处理开发包的问题上有点差别.JavaWeb不能将mysql-connector-java-5.1.7-bin.ja ...
- 【JavaWeb】JDBC连接MySQL数据库
正文之前 在之前写的JavaWeb项目中使用了JDBC,在此来回顾一下,并做个demo看看,先来看看JDBC的概念 Java数据库连接,(Java Database Connectivity,简称JD ...
- JavaWeb用Jdbc操作MySql数据库(二)
一.仍然使用前面的环境和示例数据库. 二.建立发出注册请求的页面index3.jsp. <%@ page language="java" import="java. ...
- JavaWeb 例子 JDBC+JSP登陆注册留言板
注册页面: <%@ page language="java" contentType="text/html; charset=UTF-8" pageEnc ...
- JavaWeb基础—JDBC(二)事务与批处理
一.批处理 这里给出PrepareStatement的示例,优点是可以发送预编译的SQL,缺点是SQL语句无法更换,但参数可以更换 批处理:多条语句的处理 mysql默认是关闭的,要打开需要在url后 ...
- JavaWeb基础—JDBC入门
一.什么是JDBC JDBC全称为:Java Data Base Connectivity(java数据库连接),它主要由接口组成 二.JDBC原理概述 JDBC原理:其实就是一组规范(就是对类的规范 ...
- Javaweb入门 JDBC第一天
JDBC的定义和作用 DBC(Java DataBase Connectivity) Java数据库连接, 其实就是利用Java语言/程序连接并访问数据库的一门技术. 之前我们可以通过cmd或者nav ...
- javaweb学习——JDBC(五)
管理结果集 JDBC使用ResultSet来封装查询到的结果集,然后移动记录指针来取出结果集的内容,除此之外,JDBC还允许通过ResultSet来更新记录,并提供了ResultSetMetaData ...
- JDBC链接MySQL
首先,这里的JavaWeb使用JDBC的方法与Java的使用方法相似,但是有不同之处: 在Java中导入驱动包以后,直接用DriverManager.getConnection()获取连接对象, 而在 ...
随机推荐
- RabbitMQ确认机制问题处理
现象: 手动在后台创建两个消息反馈队列 代码中监听到消息队列后,对消息进行处理并确认,代码为: 运行代码后,消息未从队列扔出去. 原因及解决方案:后台手动创建队列后,在监听消息中又对队列进行声明创建, ...
- iOS 设置#ffff 这种颜色
UI给图的时候给的是#f2f2f2 让我设置.没有你要的rgb. 所以只能自行解决封装了代码 HexColors.h #import "TargetConditionals.h" ...
- java学习书籍推荐
1. Java 语言基础 谈到Java 语言基础学习的书籍,大家肯定会推荐Bruce Eckel 的<Thinking in Java >.它是一本写的相当深刻的技术书籍,Java 语言基 ...
- 核心动画(UIView封装动画)
一.UIView动画(首尾) 1.简单说明 UIKit直接将动画集成到UIView类中,当内部的一些属性发生改变时,UIView将为这些改变提供动画支持 执行动画所需要的工作由UIView类自动完成, ...
- rips中如何使用PHP虚拟机自带函数--token_get_all
这两天在看rips源码,发现,它在审计php代码时调用了php虚拟机自带的token_get_all此函数. 这一函数会将php源码按照内置的规则进行归纳,并输出成数组格式. 如: <?php ...
- EFcore与动态模型
在开发商城系统的时候,大家会遇到这样的需求,商城系统里支持多种商品类型,比如衣服,手机,首饰等,每一种产品类型都有自己独有的参数信息,比如衣服有颜色,首饰有材质等,大家可以上淘宝看一下就明白了.现在的 ...
- linux网卡配置
6.3网卡配置 DEVICE=eth0 TYPE=Ethernet BOOTPROTO=dhcp ONBOOT=yes NETMASK=255.255.255.0 GETWAY=192.168.1.2 ...
- JS中的作用域链
在js中数据的声明方式有两种: 1.用var声明,例如:var num = 10: 2.直接声明,例如:num = 10: 两种声明方式在某些情况下是有区别的: var data = 10; func ...
- Java中String类型的部分用法
1.如何将字符串转换为整型数值? int i = Integer.parseInt("20"); 2.如何用“==”还是equals比较两个字符串? “==”是用来比较俩引用是不是 ...
- VS2010 MSDN的介绍
5 cout << temp; 博客园第一篇文章的开始. VS2010 MSDN的文档介绍 https://msdn.microsoft.com/zh-cn/library/dd8 ...