学习笔记—JDBC
JDBC的概念
JDBC(Java DataBase Connectivity,java数据库连接)是一种用于执行SQL语句的Java API,可以为多种关系数据库提供统一访问,它由一组用Java语言编写的类和接口组成。JDBC提供了一种基准,据此可以构建更高级的工具和接口,使数据库开发人员能够编写数据库应用程序。
JDBC的使用流程
导入jar包
1、下载有关数据库的jar包(下载地址:http://central.maven.org/maven2/mysql/mysql-connector-java/)
2、导入jar包,在Build Path里配置环境
Build Path-->Add JARs


加载驱动
// 加载驱动程序:Class.forName(driverClass) // 加载mysql驱动:
Class.forName("com.mysql.jdbc.Driver");
Class.forName("com.mysql.cj.jdbc.Driver"); // 8.0版本以后的mysql驱动 // 加载oracle驱动:
Class.forName("oracle.jdbc.driver.OracleDriver");
获取数据库连接对象
String url = "jdbc:mysql://localhost:3306/test";
String user = "root";
String password = "123456";
Connection conn = DriverManager.getConnection(url, user, password);
参数含义:
url: 表示要连接的数据地址
user: 数据库的用户名
password: 数据库的密码
作用:
连接到指定的数据库并返回连接对象。
创建sql命令对象(编译和发送sql命令给数据库)
Statement stmt = conn.createStatement();
创建sql命令
String sql = "insert into user values(13,'王13','wang13')";
指定sql命令,获得返回结果
int i = stmt.executeUpdate(sql);
ResultSet rs = stmt.executeQuery(sql)
增删改 时调用 stmt.executeUpdate(sql)方法。返回值为 int类型。大于等于 1 时表示成功,为 0 时表示失败。
查询 调用 stmt.executeQuery(sql)方法。返回ResultSet 类型,表示一个结果集。
ResultSet对象是基于指针进行数据存储的,类似枚举。不便于数据的针对性的获取。可将数据转换到ArrayList中进行存储。
关闭资源
stmt.close();
conn.close();
关闭之前打开的资源。需要抛出异常。
对数据的增删改查
增加 (insert)
// 声明jdbc变量
Connection conn = null;
Statement stmt = null;
// 声明jdbc参数
String driver = "com.mysql.cj.jdbc.Driver";
String url = "jdbc:mysql://localhost:3306/test";
String user = "root";
String password = "123456";
try {
// 1 加载驱动类
Class.forName(driver);
// 2.获取数据库连接对象(连接指定数据库)
conn = DriverManager.getConnection(url, user, password);
// 3.创建sql命令对象(编译和发送sql命令给数据库)
stmt = conn.createStatement();
// 4.创建sql命令
String sql = "insert into user values(13,'王13','wang13')";
// 5.指定sql命令
int i = stmt.executeUpdate(sql);
System.out.println("执行结果:" + i);
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
// 6.关闭资源
try {
stmt.close();
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
修改 (update)
// 声明jdbc变量
Connection conn = null;
Statement stmt = null; // 声明jdbc参数
String driver = "com.mysql.cj.jdbc.Driver";
String url = "jdbc:mysql://localhost:3306/test";
String user = "root";
String password = "123456"; try {
// 1 加载驱动类
Class.forName(driver); // 2.获取数据库连接对象(连接指定数据库)
conn = DriverManager.getConnection(url, user, password); // 3.创建sql命令对象(编译和发送sql命令给数据库)
stmt = conn.createStatement(); // 4.创建sql命令
String sql = "update user set username='王update' where id=13"; // 5.指定sql命令
int i = stmt.executeUpdate(sql);
System.out.println("执行结果:" + i); } catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
// 6.关闭资源
try {
stmt.close();
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
删除 (delete)
// 声明jdbc变量
Connection conn = null;
Statement stmt = null; // 声明jdbc参数
String driver = "com.mysql.cj.jdbc.Driver";
String url = "jdbc:mysql://localhost:3306/test";
String user = "root";
String password = "123456"; try {
// 1 加载驱动类
Class.forName(driver); // 2.获取数据库连接对象(连接指定数据库)
conn = DriverManager.getConnection(url, user, password); // 3.创建sql命令对象(编译和发送sql命令给数据库)
stmt = conn.createStatement(); // 4.创建sql命令
String sql = "delete from user where id=11"; // 5.指定sql命令
int i = stmt.executeUpdate(sql);
System.out.println("执行结果:" + i); } catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
// 6.关闭资源
try {
stmt.close();
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
查询(select)
// 声明jdbc变量
Connection conn = null;
Statement stmt = null;
// 声明jdbc参数
String driver = "com.mysql.cj.jdbc.Driver";
String url = "jdbc:mysql://localhost:3306/test";
String user = "root";
String password = "123456";
try {
// 1 加载驱动类
Class.forName(driver);
// 2.获取数据库连接对象(连接指定数据库)
conn = DriverManager.getConnection(url, user, password);
// 3.创建sql命令对象(编译和发送sql命令给数据库)
stmt = conn.createStatement();
// 4.创建sql命令
String sql = "select * from user";
// 5.指定sql命令
ResultSet rs = stmt.executeQuery(sql);
while (rs.next()) {
System.out.println(rs.getInt(1) + " " + rs.getString("username") + " " + rs.getString(3));
}
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
// 6.关闭资源
try {
stmt.close();
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
JDBC的事务管理
事务:一个事件的完成需要几个子操作的联合完成。只要有一个子操作执行失败,则数据回滚到原始状态,都成功则提交数据。
JDBC默认为自动提交事务。进行增删改操作时应将其设置为手动提交。
conn.setAutoCommit(false);
默认值为true,自动提交。 false 为手动提交。
使用try catch进行SQL命令执行提交和回滚。
try {
conn.commit();
} catch (SQLException e) {
conn.rollback();
e.printStackTrace();
}
try 中使用 conn.commit() 进行数据提交,若报错则在 catch 中使用 conn.rollback() 进行事务回滚。
使用PreparedStatement对象进行数据库操作
PreparedStatement对象与Statement对象的比较
Statement对象进行数据库操作时可能会出现 sql 注入的风险。
preparedStatement对象在sql语句中使用占位符,可以防止sql注入。可以预编译,批量执行同一条SQL语句时效率远大于Statement对象。
// statement对象执行sql语句
Statement stmt = conn.createStatement();
String sql = "insert into user values(13,'王13','wang13')";
int i = stmt.executeUpdate(sql); // preparedStatement对象执行sql语句
String sql = "insert into user values(?,?,?)";
PraparedStatement ps = conn.prepareStatement(sql);
ps.setInt(1, 15);
ps.setString(2, "王15");
ps.setString(3, "123456");
int i = ps.executeUpdate();
优化封装代码
driver=com.mysql.cj.jdbc.Driver
url=jdbc:mysql://localhost:3306/test
user=root
password=123456
db.properties
import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Properties; public class JUtil {
private static String driver;
private static String url;
private static String user;
private static String password; static {
// 创建properties对象获取属性文件的内容
Properties p = new Properties();
// 获取属性文件的读取对象流
InputStream is = JUtil.class.getResourceAsStream("/db.properties");
try {
// 加载属性配置文件
p.load(is);
// 获取jdbc参数
driver = p.getProperty("driver");
url = p.getProperty("url");
user = p.getProperty("user");
password = p.getProperty("password");
// 加载驱动
Class.forName(driver);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} // 获取Connection对象
public static Connection getConn() {
Connection conn = null;
try {
conn = DriverManager.getConnection(url, user, password);
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return conn;
} // 获取PreparedStatement对象
public static PreparedStatement getPre(Connection conn,String sql) {
PreparedStatement ps = null;
try {
ps = conn.prepareStatement(sql);
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return ps;
} // 关闭资源
public static void closeAll(Statement stmt,Connection conn) {
try {
stmt.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
try {
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
工具类Jutil
/**
* 封装DML
* @param sql sql语句
* @param objs 参数数组
* @return i 大于等于 1 时执行成功,等于 0 时执行失败。
*/
private int executeDML(String sql,Object ... objs) {
Connection conn = null;
PreparedStatement ps = null;
int i= 0;
try {
// 创建连接对象
conn = JUtil.getConn();
// 手动提交事物
conn.setAutoCommit(false);
// 获得sql语句及参数
ps = JUtil.getPre(conn, sql);
// 给占位符赋值
for(int j = 0; j < objs.length; j++) {
ps.setObject(j+1, objs[j]);
}
// 执行sql命令
i = ps.executeUpdate();
// 提交
conn.commit();
} catch (SQLException e) {
try {
// 回滚
conn.rollback();
} catch (SQLException e1) {
e1.printStackTrace();
}
} finally {
JUtil.closeAll(ps, conn);
}
return i;
}
封装DML(增删改)
/**
* 增加
* @param id
* @param name
* @param pwd
* @return i 大于等于 1 时执行成功,等于 0 时执行失败。
*/
public int insert(int id,String name,String pwd) {
String sql = "insert into user values(?,?,?)";
int i = executeDML(sql,id,name,pwd);
return i;
}
insert增加
/**
* 修改
* @param id
* @param name
* @param pwd
* @return i 大于等于 1 时执行成功,等于 0 时执行失败。
*/
public int update(int id,String name,String pwd) {
String sql = "update user set username=? where id=?";
int i = executeDML(sql,name,id);
return i;
}
update修改
/**
* 删除
* @param id
* @param name
* @param pwd
* @return i 大于等于 1 时执行成功,等于 0 时执行失败。
*/
public int delete(int id,String name,String pwd) {
String sql = "delete from user where id=?";
int i = executeDML(sql, id);
return i;
}
delete删除
/**
* 查询
* @return
*/
public List<User> select() {
Connection conn = JUtil.getConn();
String sql = "select * from user";
PreparedStatement ps = JUtil.getPre(conn, sql);
List<User> us = new ArrayList<User>();
try {
ResultSet rs = ps.executeQuery();
while(rs.next()) {
User u = new User();
u.setId(rs.getInt(1));
u.setName(rs.getString(2));
u.setPwd(rs.getString(3));
us.add(u);
}
} catch (SQLException e) {
e.printStackTrace();
} finally {
JUtil.closeAll(ps, conn);
}
return us;
}
select查询
常见的JDBC错误
1. ClassNotFoundException:
驱动类未找到
2. java.sql.SQLException: No suitable driver found for :mysql://localhost:3306/test
URL错误
3. No database selected
未找到数据库
4. Access denied for user 'root1'@'localhost' (using password: YES)
用户名或密码错误
5. java.sql.SQLSyntaxErrorException: You have an error in your SQL syntax;
sql语句错误
6. java.sql.SQLIntegrityConstraintViolationException: Duplicate entry '13' for key 'PRIMARY'
主键冲突
学习笔记—JDBC的更多相关文章
- Java学习笔记——JDBC读取properties属性文件
Java 中的 properties 文件是一种配置文件,主要用于表达配置信息,文件类型为*.properties,格式为文本文件. 文件的内容是格式是"键=值"(key-valu ...
- Java学习笔记——JDBC之与数据库MySQL的连接以及增删改查等操作
必须的准备工作 一.MySQL的安装.可以参考博文: http://blog.csdn.net/jueblog/article/details/9499245 二.下载 jdbc 驱动.可以从在官网上 ...
- Java学习笔记--JDBC数据库的使用
参考 hu_shengyang的专栏 : http://blog.csdn.net/hu_shengyang/article/details/6290029 一. JDBC API中提供的常用数据库 ...
- JAVA学习笔记 -- JDBC及其应用
一个.准备工作 1.开放SQL Server服务与支持TCP/IP 进一步确认TCPport watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdTAxMjk ...
- Java学习笔记——JDBC之PreparedStatement类中“预编译”的综合应用
预编译 SQL 语句被预编译并存储在 PreparedStatement 对象中.然后可以使用此对象多次高效地执行该语句. 预编译的优点 1.PreparedStatement是预编译的,对于批量处理 ...
- JMeter学习笔记--JDBC测试计划-连接Mysql
1.首先要下载jar包,mysql-connector-java-5.1.7-bin.jar 放到Jmeter的lib文件下ext下 2.添加JDBC Connection Configuration ...
- 学习笔记-JDBC连接数据库操作的步骤
前言 这里我就以JDBC连接数据库操作查询的步骤作以演示,有不到之处敬请批评指正! 一.jdbc连接简要步骤 1.加载驱动器. 2.创建connection对象. 3.创建Statement对象. 4 ...
- [原创]java WEB学习笔记109:Spring学习---spring对JDBC的支持:使用 JdbcTemplate 查询数据库,简化 JDBC 模板查询,在 JDBC 模板中使用具名参数两种实现
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- MySQL数据库学习笔记(十)----JDBC事务处理、封装JDBC工具类
[声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...
随机推荐
- 【JVM虚拟机】(8)--深入理解Class中--方法、属性表集合
#[JVM虚拟机](8)--深入理解Class中--方法.属性表集合 之前有关class文件已经写了两篇博客: 1.[JVM虚拟机](5)---深入理解JVM-Class中常量池 2.[JVM虚拟机] ...
- asp.net core系列 45 Web应用 模型绑定和验证
一. 模型绑定 ASP.NET Core MVC 中的模型绑定,是将 HTTP 请求中的数据映射到action方法参数. 这些参数可能是简单类型的参数,如字符串.整数或浮点数,也可能是复杂类型的参数. ...
- handler原理
一.消息机制概述 1.消息机制的简介 (1)Handler是什么 handler使Android给我们提供的用来更新UI的一套机制,也是一套消息处理机制:我们可以用它发送处理消息. (2)Androi ...
- Android6.0 源码修改之 Contacts应用
一.Contacts应用的主界面和联系人详情界面增加顶部菜单添加退出按钮 通过Hierarchy View 工具可以发现 主界面对应的类为 PeopleActivity 联系人详情界面对应的类为 Qu ...
- [PHP] sys_get_temp_dir()和tempnam()函数报错与环境变量的配置问题
1.项目运行过程中遇到个问题,保存临时文件时,一直返回false 2.根据经验这个是在/tmp目录下建立临时文件,所以检查了一遍权限问题,发现权限没有问题 3.查出sys_get_temp_dir() ...
- GIS之家小专栏
专栏简介:WebGIS开发者@GIS之家,一直混迹GIS行业,关注WebGIS开发方向,在本专栏中,分享WebGIS入门开发系列技术文章 核心内容: arcgis api 3.x for js开发系列 ...
- cesium 之三维场景展示篇(附源码下载)
前言 cesium 官网的api文档介绍地址cesium官网api,里面详细的介绍 cesium 各个类的介绍,还有就是在线例子:cesium 官网在线例子,这个也是学习 cesium 的好素材. 内 ...
- Vue番外篇 -- vue-router浅析原理
近期被问到一个问题,在你们项目中使用的是Vue的SPA(单页面)还是Vue的多页面设计? 这篇文章主要围绕Vue的SPA单页面设计展开. 关于如何展开Vue多页面设计请点击查看. 官网vue-rout ...
- 关于单元测试的思考--Asp.Net Core单元测试最佳实践
在我们码字过程中,单元测试是必不可少的.但在从业过程中,很多开发者却对单元测试望而却步.有些时候并不是不想写,而是常常会碰到下面这些问题,让开发者放下了码字的脚步: 这个类初始数据太麻烦,你看:new ...
- 最短路问题之Bellman-ford算法
题目: 最短路:给定两个顶点,在以这两个点为起点和终点的路径中,边的权值和最小的路径.考虑权值为点之间的距离. 单源最短路问题,Bellman-ford算法 思路:每次循环检查所有边,可优化. 应用于 ...