JDBC

  • ​​资源共享​​
  • ​​一、概念​​
  • ​​为什么要用JDBC连接池?​​
  • ​​二、快速入门​​
  • ​​2.1 实例​​
  • ​​2.2 详解各个对象​​
  • ​​1.DriverManager :驱动管理对象​​
  • ​​2.connection :数据库连接对象​​
  • ​​3.statement :执行sql的对象​​
  • ​​4.Resultset :结果集对象​​
  • ​​查询练习:​​
  • ​​Emp类​​
  • ​​查询方法​​
  • ​​5.Preparedstatement :执行sql的对象​​
  • ​​1. SQL注入问题​​
  • ​​解决方案​​
  • ​​三、JDBC工具类​​
  • ​​目的​​
  • ​​分析​​
  • ​​抽取一个方法释放资源​​
  • ​​使用工具类​​
  • ​​登陆练习​​
  • ​​数据库​​
  • ​​代码​​
  • ​​四、JDBC控制事务​​
  • ​​转账实例​​
  • ​​五、数据库连接池​​
  • ​​概念​​
  • ​​好处​​
  • ​​实现​​
  • ​​一、C3P0:数据库连接池技术​​
  • ​​1.导入jar包​​
  • ​​2.定义配置文件​​
  • ​​3.创建核心对象​​
  • ​​4.获取连接​​
  • ​​二、Druid:数据库连接池实现技术​​
  • ​​1.写入jar包​​
  • ​​2.定义配置文件​​
  • ​​3.获取数据库连接池对象:通过工厂类来获取​​
  • ​​4.获取连接​​
  • ​​5.实例​​
  • ​​三、Druid工具类​​
  • ​​实例​​
  • ​​测试​​
  • ​​查看数据库​​
  • ​​四、Spring JDBC​​
  • ​​概念​​
  • ​​什么是jar包?​​
  • ​​步骤​​
  • ​​1.导入jar包​​
  • ​​2.创建JdbcTemplate对象。​​
  • ​​3.调用JdbcTemplate的方法来完成CRUD的操作​​
  • ​​实例​​
  • ​​查看数据库数据​​
  • ​​DML练习​​
  • ​​代码​​
  • ​​DQL练习​​
  • ​​RowMapper的使用实例​​
  • ​​1.RowMapper的基本使用​​
  • ​​1.1 BeanPropertyRowMapper​​
  • ​​代码如下​​
  • ​​代码​​
  • ​​输出结果​​

资源共享

最新c3p0所有jar包(完整版)
百度网盘下载链接:https://pan.baidu.com/s/1o9cBkMVb_kZmAksZjjoZYg 密码:c7pr

阿里druid数据连接池jar包(完整版)
百度网盘地址:https://pan.baidu.com/s/1U9v5c_DindqXva92JeRVJQ 密码:nquq

一、概念

JDBC全程:

Java Database Connectivity java数据库连接

JDBC本质:其实是官方(SUN公司)定义的一套操作所有关系型数据库的规则,即接口。各个数据库厂商去实现这套接口,提供数据库驱动jar包。我们可以使用这套接口(JDBC)编程,真正执行的代码是驱动jar包中的实现类

为什么要用JDBC连接池?

对数据库进行频繁连接、开启和关闭操作,会造成数据库资源的浪费,十分影像数据库的性能。

**一般我们在项目中都会抽取出一些公共设置数据库连接的配置。项目在修改的时就可以很方便的去修改而不需要在无尽的代码中,去修改所有的代码。提高了软件的可重用性**

二、快速入门

步骤:

  1. 导入驱动jar包 mysql-connector-java-5.1.37-bin.jar
  1. 右键 --> Add As Library
  1. 注册驱动
  2. 获取数据库连接对象
  3. 定义sql

5.获取执行sql语句的对象statement

6.执行sql,接收返回的结果

7.处理结果

8.释放资源

jar包资源

​https://cdn.mysql.com/archives/mysql-connector-java-5.1/mysql-connector-java-5.1.37.zip)​

2.1 实例

package cn.caq.jdbc;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.Statement;

public class JdbcDemo01 {
public static void main(String[] args) throws Exception {
//1.导入驱动jar包
//2.注册驱动
Class.forName("com.mysql.jdbc.Driver");
//3.获取数据库连接对象
Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/db2", "root", "root");
//4.定义sql语句
String sql = "update account set balance = 1000 where id = 1";
//5.获取执行sql语句的对象 Statement
Statement statement = connection.createStatement();
//6.执行sql
int i = statement.executeUpdate(sql);
//7.处理结果
System.out.println(i);
//8.释放资源
statement.close();
connection.close();
}
}

2.2 详解各个对象

1.DriverManager :驱动管理对象

功能:
1.注册驱动:告诉程序该使用哪一个数据库驱动jar
static void registerDriver(Driver driver) :注册与给定的驱动程序 DriverManager

写代码使用:class.forName( “com.mysql.jdbc.Driver” );

通过查看源码发现:在com.mysql.jdbc.Driver类中存在静态代码块

​ static {
​ try {
​ java.sql.DriverManager.registerDriver(new Driver());​ }catch (SQLException E) {
​ throw new RuntimeException( “can’t register driver!”);
​ }


注意: mysq15之后的驱动jar包可以省略注册驱动的步骤。

2.获取数据库连接

搞懂JDBC这一篇就够了!!!_java

Connection conn = DriverManager.getConnection(“jdbc:mysql://localhost:3306/db2”, “root”, “root”);

url:指定连接的路径

细节∶如果连接的是本机mysql服务器,并且mysql服务默认端口是3306,则url可以简写为:jdbc:mysql///数据库名

user:用户名

password:密码

2.connection :数据库连接对象

功能:
1.获取执行sql 的对象
statement createstatement()
Preparedstatement preparestatement(string sql)

2.管理事务∶

开启事务: setAutoCommit(boolean autoCommit):调用该方法设置参数为false,即开启事务

提交事务: commit()

回滚事务:rollback()

3.statement :执行sql的对象

1.执行sql

  1. boolean execute(String sql) : 可以执行任意的sql,了解即可
  2. int executeUpdate(String sql) : 执行DML (insert、 update、 delete) 语句、DDL(create, alter、 drop)语句
    返回值:影响的行数,可以通过这个影响的行数判断DML语句是否执行成功返回值>0的则执行成功,反之,则失败。
  3. ResultSet executeQuery(String sql) : 执行DQL (select)语句

2.练习

1.添加一条记录

2.修改记录

3.删除记录

为了代码健壮性,进行捕获异常

String sql = "update account set balance = 40 where id = 1"; String sql = "insert into account values(3,'ws',1002)"; String sql = "delete from account where id = 3";

修改后的代码

package cn.caq.jdbc;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.sql.Statement;

public class JdbcDemo03 {
public static void main(String[] args) {
Connection conn = null;
Statement stmt = null;
try {
//1.注册驱动
Class.forName("com.mysql.jdbc.Driver");
//2.获取连接对象
conn = DriverManager.getConnection("jdbc:mysql:///db2", "root", "root");
//3.定义sql
String sql = "insert into account values(3,'ws',1002)";
//4.获取执行sql对象
stmt = conn.createStatement();
//5.执行sql
int count = stmt.executeUpdate(sql);
//6.处理结果
System.out.println(count);
if (count > 0){
System.out.println("修改成功");
}else {
System.out.println("修改失败");
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (SQLException throwables) {
throwables.printStackTrace();
} finally {
if (stmt != null){
try {
stmt.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

if (conn != null){
try {
conn.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

}
}
}

4.Resultset :结果集对象

结果集对象,封装查询结果

1.next();游标向下移动一行,判断当前行是否是最后一行末尾

它的返回值是布尔值

2.getXXX:获取数据,XXX代表数据类型int getInt() String getString()

...
//int :代表列的编号,从1开始 如:getString(1)
//string: 代表列的名称 如:getString("name")
int id = res.getInt(1);
String name = res.getString("name");
double balance = res.getDouble(3);
System.out.println(id + "---" + name + "---" + balance);
结果为:
1---zs---40.0

使用步骤:

  • 游标向下移动一行
  • 判断是否有数据
  • 获取数据

正确用法

while (res.next()){
int id = res.getInt(1);
String name = res.getString("name");
double balance = res.getDouble(3);
System.out.println(id + "---" + name + "---" + balance);
}
查询练习:

查询表的数据封装为对象,返回一个列表打印处理

思路:

定义一个方法,查询emp表的数据将其封装为对象,然后装载集合,返回

1.根据表的结构定义一个emp类

2.查询的结果封装为emp类的对象

3.定义方法public list findAll(){}查询表中数据并封装为集合

Emp类
package cn.caq.domain;

import java.util.Date;

/**
* 封装Emp表数据的Java Bean
*/

public class Emp {
private int id;
private String ename;
private int job_id;
private int mgr;
private Date joindate;
private double salary;
private double bounds;

private int dept_id;

public int getId() {
return id;
}


public String getEname() {
return ename;
}

public void setEname(String ename) {
this.ename = ename;
}

public void setId(int id) {
this.id = id;
}



public int getJob_id() {
return job_id;
}

public void setJob_id(int job_id) {
this.job_id = job_id;
}

public int getMgr() {
return mgr;
}

public void setMgr(int mgr) {
this.mgr = mgr;
}

public Date getJoindate() {
return joindate;
}

public void setJoindate(Date joindate) {
this.joindate = joindate;
}

public double getSalary() {
return salary;
}

public void setSalary(double salary) {
this.salary = salary;
}

public double getBounds() {
return bounds;
}

public void setBounds(double bounds) {
this.bounds = bounds;
}

public int getDept_id() {
return dept_id;
}

public void setDept_id(int dept_id) {
this.dept_id = dept_id;
}

@Override
public String toString() {
return "Emp{" +
"id=" + id +
", ename='" + ename + '\'' +
", job_id=" + job_id +
'}';
}
}
查询方法
package cn.caq.jdbc;

import cn.caq.domain.Emp;

import java.sql.*;
import java.util.ArrayList;
import java.util.List;


/**
* 定义一个方法,查询emp表的数据将其封装为对象,然后装载集合,返回
*/

public class JdbcDemo06 {

public static void main(String[]args){
List<Emp> list = new JdbcDemo06().findall();
System.out.println(list);
}

public List<Emp> findall() {
Connection conn = null;
Statement stmt = null;
ResultSet res = null;
ArrayList<Emp> list = null;
try {
//1.注册驱动
Class.forName("com.mysql.jdbc.Driver");
//2.获取连接对象
conn = DriverManager.getConnection("jdbc:mysql:///db2", "root", "root");
//3.定义sql
String sql = "select * from emp";
//4.获取执行sql对象
stmt = conn.createStatement();
//5.执行sql
res = stmt.executeQuery(sql);
//6.处理结果
Emp emp = null;
list = new ArrayList<>();

while (res.next()) {
int id = res.getInt("id");
String ename = res.getString("ename");
int job_id = res.getInt("job_id");

//创建emp对象,并赋值
emp = new Emp();
emp.setId(id);
emp.setEname(ename);
emp.setJob_id(job_id);
//装载集合
list.add(emp);

}
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (SQLException throwables) {
throwables.printStackTrace();
} finally {
if (stmt != null) {
try {
stmt.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

if (conn != null) {
try {
conn.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

if (res != null) {
try {
res.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

}
return list;
}
}

输出结果为:
[Emp{id=1001, ename='孙悟空', job_id=4}, Emp{id=1002, ename='卢俊义', job_id=3}, Emp{id=1003, ename='林冲', job_id=3}, Emp{id=1004, ename='唐僧', job_id=2}, Emp{id=1005, ename='李逵', job_id=4}, Emp{id=1006, ename='宋江', job_id=2}, Emp{id=1007, ename='刘备', job_id=2}, Emp{id=1008, ename='猪八戒', job_id=4}, Emp{id=1009, ename='罗贯中', job_id=1}, Emp{id=1010, ename='吴用', job_id=3}, Emp{id=1011, ename='沙僧', job_id=4}, Emp{id=1012, ename='李逵', job_id=4}, Emp{id=1013, ename='小白龙', job_id=4}, Emp{id=1014, ename='关羽', job_id=4}]

5.Preparedstatement :执行sql的对象

1. SQL注入问题

用户随便输入密码:a’ or ‘a’ = 'a

String sql = "select * from user where username = '"+username+"' and password = '"+password+"'";
System.out.println(sql);

拼接后的sql:select * from user where username = ‘dafsdfda’ and password = ‘a’ or ‘a’ = 'a’

请输入username:
fassag
请输入password
a’ or ‘a’ = 'a
Success!!!

解决方案

通过Preparedstatement :执行sql

它是预编译的SQL( 一次编译、多次运行,省去了解析优化等过程,此外预编译语句能防止sql注入 。)

参数使用?作为占位符

全部的流程为:

步骠:
1.导入驱动jar包mysql-connector-java-5.1.37-bin.jar

2.注册驱动

3.获取数据库连接对象connection

4.定义sql
注意:sql的参数使用?作为占位符。如: select * from user where username ? and password ?;

5.获取执行sql语句的对象preparedstatement Connection.preparestatement(string sql)
6.给?赋值:
*方法: setXXX(参数1,参数2)
参数1:?的位置编号 从1开始

参数2:?的值

7、执行sql,接受返回结果,不需要传递sql语句

8.处理结果

9.释放资源

三、JDBC工具类

目的

简化书写

分析

1.注册驱动也抽取

2.抽取一个方法获取连接对象

需求:不想传递参数(麻烦),还带保证工具类的通用新

解决:properties配置文件

url=jdbc:mysql:///db2
user=root
password=root
driver=com.mysql.jdbc.Driver


抽取一个方法释放资源

package cn.caq.utils;

import java.io.FileReader;
import java.io.IOException;
import java.net.URL;
import java.sql.*;
import java.util.Properties;

/**
* JDBC工具类
*/
public class JDBCutils {
private static String url;
private static String user;
private static String password;
private static String driver;

/**
* 文件的读取,只需要读取一次即可拿到这些值。使用静态代码块
*/
static {
try {
//读取资源文件,获取值
//1.创建Properties集合类
Properties pro = new Properties();
//获取src路径下的文件方式--->ClassLoader 类加载器
ClassLoader classLoader = JDBCutils.class.getClassLoader();
URL res = classLoader.getResource("jdbc.properties");//得到文件的绝对路径
String path = res.getPath();//将绝对路径转化为字符串形式
//FileReader用来读用户本地的文件
pro.load(new FileReader(path));

//2.加载文件
//pro.load(new FileReader("src/jdbc.properties"));
//3.获取数据,赋值
url = pro.getProperty("url");
user = pro.getProperty("user");
password = pro.getProperty("password");
driver = pro.getProperty("driver");
//4.注册驱动
Class.forName(driver);
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}

/**
* 获取连接的方法
*/
public static Connection getConnection() throws SQLException {
return DriverManager.getConnection(url, user, password);
}

/**
* 释放资源的方法
*/

public static void close(ResultSet rs, Statement stmt, Connection conn) {
if (rs != null) {
try {
rs.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

if (stmt != null) {
try {
stmt.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

if (conn != null) {
try {
conn.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
}
}

使用工具类

package cn.caq.jdbc;

import cn.caq.domain.Emp;
import cn.caq.utils.JDBCutils;

import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.List;

public class JdbcDemo07 {

public static void main(String[] args) {
List<Emp> list = new JdbcDemo07().findall2();
System.out.println(list);
}

/**
* 演示JDBC工具类
*/
public List<Emp> findall2() {
Connection conn = null;
Statement stmt = null;
ResultSet res = null;
ArrayList<Emp> list = null;
try {
// //1.注册驱动
// Class.forName("com.mysql.jdbc.Driver");
// //2.获取连接对象
// conn = DriverManager.getConnection("jdbc:mysql:///db2", "root", "root");
conn = JDBCutils.getConnection();
//3.定义sql
String sql = "select * from emp";
//4.获取执行sql对象
stmt = conn.createStatement();
//5.执行sql
res = stmt.executeQuery(sql);
//6.处理结果
Emp emp = null;
list = new ArrayList<>();

while (res.next()) {
int id = res.getInt("id");
String ename = res.getString("ename");
int job_id = res.getInt("job_id");

//创建emp对象,并赋值
emp = new Emp();
emp.setId(id);
emp.setEname(ename);
emp.setJob_id(job_id);
//装载集合
list.add(emp);

}
} catch (SQLException throwables) {
throwables.printStackTrace();
} finally {
JDBCutils.close(res,stmt,conn);
}
return list;
}
}

登陆练习

需求:

1.通过键盘录入用户名和密码

2.判断用户是否登录成功

select * from user where username = “” and password = “”;

如果这个sql有查询结果,则成功反之失败

数据库

搞懂JDBC这一篇就够了!!!_开发语言_02

更改配置文件即可

搞懂JDBC这一篇就够了!!!_开发语言_03

代码

package cn.caq.jdbc;

import cn.caq.utils.JDBCutils;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Scanner;

/**
* 需求:
* 1.通过键盘录入用户名和密码
* 2.判断用户是否登录成功
*/

public class JDBCDemo08 {

public static void main(String[]args){
//1.键盘录入,接受用户名和密码
Scanner scanner = new Scanner(System.in);
System.out.println("请输入username:");
String username = scanner.nextLine();
System.out.println("请输入password");
String password = scanner.nextLine();

//2.调用方法
// JDBCDemo08 jdbcDemo08 = new JDBCDemo08();
// boolean login = jdbcDemo08.login(username, password);
boolean flag = new JDBCDemo08().login(username, password);

//3.判断结果,输出不同语句
if (flag){
System.out.println("Success!!!!!!");
}else {
System.out.println("username or passwd False!!!!!!!!");
}

}

//登录方法
public boolean login(String username,String password){
if (username == null || password == null){
return false;
}
Connection conn = null;
Statement stmt = null;
ResultSet rs = null;
//1.连接数据库
try {
conn = JDBCutils.getConnection();
//2.定义sql
String sql = "select * from user where username = '"+username+"' and password = '"+password+"'";
//3.获取执行sql的对象
stmt = conn.createStatement();
//4.执行sql语句
rs = stmt.executeQuery(sql);
//5.判断
return rs.next();//有下一行返回true
} catch (SQLException throwables) {
throwables.printStackTrace();
} finally {
JDBCutils.close(rs,stmt,conn);
}

return false;//如果出现异常返回false
}
}

输出结果为:
请输入username:
zs
请输入password
123
Success!!!!!!


请输入username:
sadfa
请输入password
24323
username or passwd False!!!!!!!!

四、JDBC控制事务

1.事务:一个包含多个步骤的业务操作。如果这个业务操作被事务管理,则这多个步骤要么同时成功,要么同时失败。

2.操作:

1.开后事务

2.提交事务

3.回滚事务

3.使用connection对象来管理事务

开启事务: setAutoCommit(boolean autoCommit) :调用该方法设置参数为false,即开后事务

在执行sql之前开启事务

提交事务: commit()

当所有sql都执行完提交事务

回滚事务: rollback()

在catch中回滚事务

转账实例

package cn.caq.jdbc;

import cn.caq.utils.JDBCutils;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;

/**
* 事务操作
*/

public class JDBCDemo10 {
public static void main(String[] args) {
Connection conn = null;
PreparedStatement pstmt1 = null;
PreparedStatement pstmt2 = null;
//1.获取连接
try {
conn = JDBCutils.getConnection();
//开启事务
conn.setAutoCommit(false);

//2.定义sql
//2.1张三 - 500
String sql1 = "update account set balance = balance - ? where id = ?";
//2.1李四 + 500
String sql2 = "update account set balance = balance + ? where id = ?";
//3.获取执行sql对象
pstmt1 = conn.prepareStatement(sql1);
pstmt2 = conn.prepareStatement(sql2);
//4.设置参数
pstmt1.setDouble(1, 500);
pstmt1.setInt(2, 1);

pstmt2.setDouble(1, 500);
pstmt2.setInt(2, 2);

//5.执行sql
pstmt1.executeUpdate();

//手动制造异常
int i = 3 / 0;

pstmt2.executeUpdate();

//提交事务
conn.commit();

} catch (Exception throwables) {
try {
if (conn != null) {
conn.rollback();
}
} catch (SQLException e) {
e.printStackTrace();
}
throwables.printStackTrace();
}
}
}


如果中途出现任何异常,则触发回滚操作!
输出结果为:
java.lang.ArithmeticException: / by zero
at cn.caq.jdbc.JDBCDemo10.main(JDBCDemo10.java:43)

金额没有发生变化

搞懂JDBC这一篇就够了!!!_java_04

五、数据库连接池

概念

其实就是一个容器(集合),存放数据库连接的容器

当系统初始化好后,容器被初始化,容器中会申请一些连接对象,当用户来访问数据库时,从容器中获取连接对象,用户访问完后,会将连接对象返回给容器

好处

节约资源

高效

实现

标准的接口:DataSource javax.sql包下的

方法:

获取连接:getConnection()

归还连接:Connection.close().如果连接对象Connection是从连接池中获取的,那么调用Connection.close()不在是关闭连接,而是归还连接

2.这个接口,一般不由人为实现,而是有数据库厂商来实现

C3P0:数据库连接池技术

Druid:数据库连接池实现技术,由阿里提供

一、C3P0:数据库连接池技术

步骤:

1.导入jar包

c3p0-0.9.5.2.jar

mchange-commons-java-0.2.12.jar

还有驱动jar包

2.定义配置文件

名称:c3p0.properties或者c3p0-config.xml

路径:直接将文件放在src目录下即可

3.创建核心对象

数据库连接池对象 ComboPoolDataSource

4.获取连接

getConnection

了解即可

二、Druid:数据库连接池实现技术

1.写入jar包

druid-1.e.9.jar

2.定义配置文件

是properties形式的(properties 性能)
可以叫任意名称,可以放在任意目录下

3.获取数据库连接池对象:通过工厂类来获取

DruidDataSourceFactory

4.获取连接

getConnection

5.实例

package com.caq.datasource.druid;
import com.alibaba.druid.pool.DruidDataSourceFactory;

import javax.sql.DataSource;
import java.io.InputStream;
import java.sql.Connection;
import java.util.Properties;

public class DruidDemo01 {
public static void main(String[] args) throws Exception {
//1.导入Jar包
//2.定义配置文件
//3.加载配置文件
Properties pro = new Properties();
InputStream is = DruidDemo01.class.getClassLoader().getResourceAsStream("druid.properties");//得到输出流
pro.load(is);//获取文件的路径
//4.获取连接池对象
DataSource ds = DruidDataSourceFactory.createDataSource(pro);
//5.获取连接
Connection conn = ds.getConnection();
System.out.println(conn);
}
}

输出结果为:
十月 20, 2021 10:32:02 下午 com.alibaba.druid.pool.DruidAbstractDataSource error
严重: maxIdle is deprecated
十月 20, 2021 10:32:03 下午 com.alibaba.druid.pool.DruidDataSource info
信息: {dataSource-1} inited
com.mysql.jdbc.JDBC4Connection@6956de9

三、Druid工具类

1.定义一个类JDBCUtils

2.提供静态代码块加载配置文件,初始化连接池对象

3.提供方法

1.获取连接方法:通过数据库连接池获取

2.释放资源

3.获取连接池的方法

实例

详细的注释都写在代码中了!

package com.caq.datasource.utils;

import com.alibaba.druid.pool.DruidDataSourceFactory;

import javax.sql.DataSource;
import java.io.IOException;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Properties;

/**
* Druid连接池工具类
*/
public class JDBCUtils {
//1.定义成员变量 DataSource
private static DataSource ds;

static {
try {
//1.加载配置文件
Properties pro = new Properties();
pro.load(JDBCUtils.class.getClassLoader().getResourceAsStream("druid.properties"));
//2.获取DataSource
ds = DruidDataSourceFactory.createDataSource(pro);

} catch (IOException e) {
e.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
}
}



/**
* 获取连接
*/
public static Connection getConnection() throws SQLException {
return ds.getConnection();
}


/**
* 释放资源
*/
//第一种情况,执行DML语句,需要释放connection数据库连接对象和statementsql语句执行对象
public static void close(Statement stmt, Connection conn) {
if (stmt != null) {
try {
stmt.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

if (conn != null) {
try {
conn.close(); //归还连接,不在是释放了
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

}

//第二种情况,执行DQL语句,释放三个资源,conn,stmt和resultSet结果集对象
//这里写一个方法重载即可
public static void close(ResultSet rs, Statement stmt, Connection conn) {
if (stmt != null) {
try {
stmt.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

if (conn != null) {
try {
conn.close(); //归还连接,不在是释放了
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
}
/**
* 获取连接池的方法
*/
public static DataSource getDataSource(){
return ds;
}

}

测试

package com.caq.datasource.druid;

import com.caq.datasource.utils.JDBCUtils;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;

/**
* 使用新的工具类
*/
public class DruidDemo02 {
public static void main(String[] args) {
/*
完成添加的操作 给account表添加一条记录
*/
Connection conn = null;
PreparedStatement pstmt = null;
try {
//1.获取连接
conn = JDBCUtils.getConnection();
//2.定义sql
String sql = "insert into account values(null,?,?)";
//3.获取pstmt对象
pstmt = conn.prepareStatement(sql);
//4.给sql赋值
pstmt.setString(1, "王五");//给第一个参数赋值为王五
pstmt.setDouble(2, 3000);//给第二个参数赋值为3000
//5.执行sql
int count = pstmt.executeUpdate();
System.out.println(count);
} catch (SQLException throwables) {
throwables.printStackTrace();
} finally {
//6.释放资源
JDBCUtils.close(pstmt,conn);
}
}
}

输出结果为
十月 21, 2021 8:40:18 下午 com.alibaba.druid.pool.DruidAbstractDataSource error
严重: maxIdle is deprecated
十月 21, 2021 8:40:19 下午 com.alibaba.druid.pool.DruidDataSource info
信息: {dataSource-1} inited
1

查看数据库

搞懂JDBC这一篇就够了!!!_开发语言_05

人直接傻掉,太强了,太tmd的好用了啊!!!!

四、Spring JDBC

概念

Spring框架对JDBC的简单封装。提供了一个JDBCTemplate对象简化JDBC的开发

什么是jar包?

学了这么久要是不知道什么是jar包那就尴尬了啊

jar包就是别人已经写好的一些类,然后将这些类进行打包,你可以将这些jar包引入你的项目中,然后就可以直接使用这些jar包中的类和属性以及方法。 so ga!!!

没错我现在才知道!!![🤦‍]

步骤

1.导入jar包
2.创建JdbcTemplate对象。

它依赖于数据源DataSource

JdbcTemplate template = new JdbcTemplate(ds);

3.调用JdbcTemplate的方法来完成CRUD的操作
  • update():执行DML语句
  • queryforMap():查询结果将结果集封装为map集合,将列名作为key,将值作为value,将这条记录封装为一个map集合
  • 注意:这个方法查询的结果集长度只能是1**
  • queryforList():查询结果将结果集封装为list集合
  • 注意:将每一条记录封装为一个Map集合,再将Map集合装载到List集合中
  • query():查询结果,将结果封装为JavaBean对象
  • query的参数:RowMapper
    一般我们使用BeanPropertyRowMapper实现类。可以完成数据到JavaBean的自动封装
  • new BeanPropertyRowiMapper<类型>(类型.class)
  • queryForObject:查询结果,将结果封装为对象
  • 一般用于聚合函数的查询

实例

package com.caq.datasource.jdbctemplate;

import com.caq.datasource.utils.JDBCUtils;
import org.springframework.jdbc.core.JdbcTemplate;

public class JdbctemplateDemo01 {
public static void main(String[] args) {
//1.导入jar包
//2.创建jdbctemplate对象
JdbcTemplate template = new JdbcTemplate(JDBCUtils.getDataSource());
//3.调用方法
String sql = "update account set balance = 5000 where id = ?";
int count = template.update(sql, 3);
System.out.println(count);
}
}

注意这个方法的使用,第一个参数是sql语句,第二个是经过preparedstatement封装后的sql语句中的?的值

搞懂JDBC这一篇就够了!!!_数据库_06

查看数据库数据

搞懂JDBC这一篇就够了!!!_sql_07

DML练习

需求:

  1. 修改1号数据的salary 为1000
  2. 添加一条记录
  3. 删除刚才添加的记录
  4. 查询id为1的记录,将其封装为Map集合
  5. 查询所有记录,将其封装为List
  6. 查询所有记录,将其封装为Emp对象的List集合
  7. 查询总记录数
代码


package com.caq.datasource.jdbctemplate;
import com.caq.datasource.utils.JDBCUtils;
import org.junit.Test;
import org.springframework.jdbc.core.JdbcTemplate;

public class JdbctemplateDemo02 {
//Junit单元测试,可以让方法独立执行
//1.获取JDBCTemplate对象
private JdbcTemplate template = new JdbcTemplate(JDBCUtils.getDataSource());


/**
* 1.修改1号数据的salary为10000
*/
@Test
public void test1(){
//2.定义sql
String sql = "update emp set salary = 10000 where id = 1001";
//3.执行sql
int count = template.update(sql);
System.out.println(count);
}

/**
* 添加一条记录
*/
@Test
public void test2(){
String sql = "insert into emp(id,ename,dept_id) values(?,?,?)";//双引只能嵌套单引
int count = template.update(sql, 1015, "gj", 10);
System.out.println(count);
}

/**
* 删除刚才添加的记录
*/
@Test
public void test3(){
String sql = "delete from emp where id = ?";
int count = template.update(sql, 1015);
System.out.println(count);
}

}

分别对三个单元进行测试

结果如下:

搞懂JDBC这一篇就够了!!!_开发语言_08

搞懂JDBC这一篇就够了!!!_sql_09

DQL练习

JavaBeans是Java中一种特殊的类,可以将多个对象封装到一个对象(bean)中。 特点是可序列化,提供无参构造器,提供getter方法和setter方法访问对象的属性

RowMapper的使用实例

1.RowMapper的基本使用

从数据库查询出来的记录全都被保存在ResultSet结果集中,我们需要将结果集中的数据一条条地获取并设置到具体的实体类上,如此,该实体类才能在接下来的程序中使用。然而问题是,每次都要这么操作实在是太麻烦了,Spring就不应该提供什么功能来替我们做这些事情吗?RowMapper就是实现这个功能的

1.1 BeanPropertyRowMapper

当查询数据库返回的是多列数据,且你需要将这些多列数据映射到某个具体的实体类上

代码如下
@Override
public Student getStudentByName2(String name) {
String sql = "select name, gender from test_student where name = ?";
return this.jdbcTemplate.queryForObject(sql, new Object[]{name},
new BeanPropertyRowMapper<>(Student.class));
}

@Override
public List<Student> getStudentsByName2(String name) {
String sql = "select name, gender from test_student where name = ?";
return this.jdbcTemplate.query(sql, new Object[]{name},
new BeanPropertyRowMapper<>(Student.class));
}
代码
package com.caq.datasource.jdbctemplate;
import com.caq.datasource.utils.JDBCUtils;
import com.caq.datasource.domain.Emp;
import org.junit.Test;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;
import java.util.List;
import java.util.Map;

public class JdbctemplateDemo02 {
//Junit单元测试,可以让方法独立执行
//1.获取JDBCTemplate对象
private JdbcTemplate template = new JdbcTemplate(JDBCUtils.getDataSource());


/**
* 1.修改1号数据的salary为10000
*/
@Test
public void test1(){
//2.定义sql
String sql = "update emp set salary = 10000 where id = 1001";
//3.执行sql
int count = template.update(sql);
System.out.println(count);
}

/**
* 添加一条记录
*/
@Test
public void test2(){
String sql = "insert into emp(id,ename,dept_id) values(?,?,?)";//双引只能嵌套单引
int count = template.update(sql, 1015, "gj", 10);
System.out.println(count);
}

/**
* 删除刚才添加的记录
*/
@Test
public void test3(){
String sql = "delete from emp where id = ?";
int count = template.update(sql, 1015);
System.out.println(count);
}

/**
* 4.查询id为1001的记录,将其封装为Map集合
* 注意:这个方法查询的结果集长度只能是一
*/
@Test
public void test4(){
String sql = "select * from emp where id = ?";
Map<String, Object> map = template.queryForMap(sql, 1001);
System.out.println(map);
//{id=1001, ename=孙悟空, job_id=4, mgr=1004, joindate=2000-12-17, salary=10000.00, bonus=null, dept_id=20}
}

/**
* 5.查询所有记录,将其封装为List
*/
@Test
public void test5(){
String sql = "select * from emp";
List<Map<String, Object>> list = template.queryForList(sql);
//iter快捷键可快速生成循环
for (Map<String, Object> stringObjectMap : list) {
System.out.println(stringObjectMap);
}
}

/**
* 6.查询所有记录,将其封装为Emp对象的List集合
*/

@Test
public void test6(){
String sql = "select * from emp";
//可以自己实现接口,也可以用别人实现号的类BeanPropertyRowMapper
List<Emp> list = template.query(sql, new BeanPropertyRowMapper<Emp>(Emp.class));
for (Emp emp:list){
System.out.println(emp);
}
}

/**
* 7.查询总记录数
*/

@Test
public void test7(){
String sql = "select count(id) from emp";
Long total = template.queryForObject(sql, Long.class);
System.out.println(total);
//14
}
}
输出结果
com.caq.datasource.jdbctemplate.JdbctemplateDemo02,test6
十月 21, 2021 10:51:47 下午 com.alibaba.druid.pool.DruidAbstractDataSource error
严重: maxIdle is deprecated
十月 21, 2021 10:51:48 下午 com.alibaba.druid.pool.DruidDataSource info
信息: {dataSource-1} inited
Emp{id=1001, ename='孙悟空', job_id=4, mgr=1004, joindate=2000-12-17 00:00:00.0, salary=10000.0, bounds=null, dept_id=20}
Emp{id=1002, ename='卢俊义', job_id=3, mgr=1006, joindate=2001-02-20 00:00:00.0, salary=16000.0, bounds=null, dept_id=30}
Emp{id=1003, ename='林冲', job_id=3, mgr=1006, joindate=2001-02-22 00:00:00.0, salary=12500.0, bounds=null, dept_id=30}
Emp{id=1004, ename='唐僧', job_id=2, mgr=1009, joindate=2001-04-02 00:00:00.0, salary=29750.0, bounds=null, dept_id=20}
Emp{id=1005, ename='李逵', job_id=4, mgr=1006, joindate=2001-09-28 00:00:00.0, salary=12500.0, bounds=null, dept_id=30}
Emp{id=1006, ename='宋江', job_id=2, mgr=1009, joindate=2001-05-01 00:00:00.0, salary=28500.0, bounds=null, dept_id=30}
Emp{id=1007, ename='刘备', job_id=2, mgr=1009, joindate=2001-09-01 00:00:00.0, salary=24500.0, bounds=null, dept_id=10}
Emp{id=1008, ename='猪八戒', job_id=4, mgr=1004, joindate=2007-04-19 00:00:00.0, salary=30000.0, bounds=null, dept_id=20}
Emp{id=1009, ename='罗贯中', job_id=1, mgr=null, joindate=2001-11-17 00:00:00.0, salary=50000.0, bounds=null, dept_id=10}
Emp{id=1010, ename='吴用', job_id=3, mgr=1006, joindate=2001-09-08 00:00:00.0, salary=15000.0, bounds=null, dept_id=30}
Emp{id=1011, ename='沙僧', job_id=4, mgr=1004, joindate=2007-05-23 00:00:00.0, salary=11000.0, bounds=null, dept_id=20}
Emp{id=1012, ename='李逵', job_id=4, mgr=1006, joindate=2001-12-03 00:00:00.0, salary=9500.0, bounds=null, dept_id=30}
Emp{id=1013, ename='小白龙', job_id=4, mgr=1004, joindate=2001-12-03 00:00:00.0, salary=30000.0, bounds=null, dept_id=20}
Emp{id=1014, ename='关羽', job_id=4, mgr=1007, joindate=2002-01-23 00:00:00.0, salary=13000.0, bounds=null, dept_id=10}

Process finished with exit code 0