JDBC03-批处理、连接池、DBUtils、事物、DAO通用方法

news2025/1/22 12:50:46

1. 封装 JDBCUtils 【关闭、得到连接】

1.1 说明

在这里插入图片描述
在这里插入图片描述

1.2 代码实现

工具类 JDBCUtils

package com.hspedu.jdbc.utils;

import java.io.FileInputStream;
import java.io.IOException;
import java.sql.*;
import java.util.Properties;
/**
* 这是一个工具类,完成 mysql 的连接和关闭资源
*/
public class JDBCUtils {

	//定义相关的属性(4 个), 因为只需要一份,因此,我们做出 static
	private static String user; //用户名
	private static String password; //密码
	private static String url; //url
	private static String driver; //驱动名
	
	//在 static 代码块去初始化
	static {
		try {
			Properties properties = new Properties();
			properties.load(new FileInputStream("src\\mysql.properties"));
			//读取相关的属性值
			user = properties.getProperty("user");
			password = properties.getProperty("password");
			url = properties.getProperty("url");
			driver = properties.getProperty("driver");
		} catch (IOException e) {
			//在实际开发中,我们可以这样处理
			//1. 将编译异常转成 运行异常
			//2. 调用者,可以选择捕获该异常,也可以选择默认处理该异常,比较方便. 
			throw new RuntimeException(e);
		} 
	}
	
	//连接数据库, 返回 Connection
	public static Connection getConnection() {
		try {
			return DriverManager.getConnection(url, user, password);
		} catch (SQLException e) {
			//1. 将编译异常转成 运行异常
			//2. 调用者,可以选择捕获该异常,也可以选择默认处理该异常,比较方便. throw new RuntimeException(e);
		}
	}
	
	//关闭相关资源
	/*
		1. ResultSet 结果集
		2. Statement 或者 PreparedStatement
		3. Connection
		4. 如果需要关闭资源,就传入对象,否则传入 null
	*/
	public static void close(ResultSet set, Statement statement, Connection connection) {
		//判断是否为 null
		try {
			if (set != null) {
				set.close();
			}
			if (statement != null) {
				statement.close();
			}
			if (connection != null) {
				connection.close();
			}
		} catch (SQLException e) {
			//将编译异常转成运行异常抛出
			throw new RuntimeException(e);
		}
	}
}

测试类

package com.hspedu.jdbc.utils;

import org.junit.jupiter.api.Test;
import java.sql.*;

/**
* 该类演示如何使用 JDBCUtils 工具类,完成 dml 和 select
*/
public class JDBCUtils_Use {

	@Test
	public void testSelect() {
		//1. 得到连接
		Connection connection = null;
		
		//2. 组织一个 sql
		String sql = "select * from actor where id = ?";
		PreparedStatement preparedStatement = null;
		ResultSet set = null;
		
		//3. 创建 PreparedStatement 对象
		try {
			connection = JDBCUtils.getConnection();
			System.out.println(connection.getClass()); //com.mysql.jdbc.JDBC4Connection
			preparedStatement = connection.prepareStatement(sql);
			preparedStatement.setInt(1, 5);//给?号赋值
			//执行, 得到结果集
			set = preparedStatement.executeQuery();
			//遍历该结果集
			while (set.next()) {
				int id = set.getInt("id");
				String name = set.getString("name");
				String sex = set.getString("sex");
				Date borndate = set.getDate("borndate");
				String phone = set.getString("phone");
				System.out.println(id + "\t" + name + "\t" + sex + "\t" + borndate + "\t" + phone);
			}
		} catch (SQLException e) {
			e.printStackTrace();
		} finally {
			//关闭资源
			JDBCUtils.close(set, preparedStatement, connection);
		}
	}
	@Test
	public void testDML() {//insert , update, delete
		//1. 得到连接
		Connection connection = null;
		
		//2. 组织一个 sql
		String sql = "update actor set name = ? where id = ?";
		
		// 测试 delete 和 insert ,自己玩. PreparedStatement preparedStatement = null;
		//3. 创建 PreparedStatement 对象
		try {
			connection = JDBCUtils.getConnection();
			preparedStatement = connection.prepareStatement(sql);
			//给占位符赋值
			preparedStatement.setString(1, "周星驰");
			preparedStatement.setInt(2, 4);
			//执行
			preparedStatement.executeUpdate();
		} catch (SQLException e) {
			e.printStackTrace();
		} finally {
			//关闭资源
			JDBCUtils.close(null, preparedStatement, connection);
		}
	}
}

2. 事务

2.1 基本介绍

在这里插入图片描述

2.2 应用实例

模拟经典的转账业务
在这里插入图片描述
在这里插入图片描述

2.3 不使用事务可能出现的问题模拟-模拟经典的转账业务

//没有使用事务. 
@Test
public void noTransaction() {
	//操作转账的业务
	//1. 得到连接
	Connection connection = null;
	
	//2. 组织一个 sql
	String sql = "update account set balance = balance - 100 where id = 1";
	String sql2 = "update account set balance = balance + 100 where id = 2";
	PreparedStatement preparedStatement = null;
	//3. 创建 PreparedStatement 对象
	try {
		connection = JDBCUtils.getConnection(); // 在默认情况下,connection 是默认自动提交
		preparedStatement = connection.prepareStatement(sql);
		preparedStatement.executeUpdate(); // 执行第 1 条 sql
		int i = 1 / 0; //抛出异常
		preparedStatement = connection.prepareStatement(sql2);
		preparedStatement.executeUpdate(); // 执行第 3 条 sql
	} catch (SQLException e) {
		e.printStackTrace();
	} finally {
		//关闭资源
		JDBCUtils.close(null, preparedStatement, connection);
	}
}

2.4 使用事务解决上述问题-模拟经典的转账业务

//事务来解决
@Test
public void useTransaction() {
	//操作转账的业务
	//1. 得到连接
	Connection connection = null;
	
	//2. 组织一个 sql
	String sql = "update account set balance = balance - 100 where id = 1";
	String sql2 = "update account set balance = balance + 100 where id = 2";
	PreparedStatement preparedStatement = null;
	//3. 创建 PreparedStatement 对象
	try {
		connection = JDBCUtils.getConnection(); // 在默认情况下,connection 是默认自动提交
		//将 connection 设置为不自动提交
		connection.setAutoCommit(false); //开启了事务
		preparedStatement = connection.prepareStatement(sql);
		preparedStatement.executeUpdate(); // 执行第 1 条 sql
		int i = 1 / 0; //抛出异常
		preparedStatement = connection.prepareStatement(sql2);
		preparedStatement.executeUpdate(); // 执行第 3 条 sql
		
		//这里提交事务
		connection.commit();
	} catch (SQLException e) {
	
		//这里我们可以进行回滚,即撤销执行的 SQL
		//默认回滚到事务开始的状态. System.out.println("执行发生了异常,撤销执行的 sql");
		try {
			connection.rollback();
		} catch (SQLException throwables) {
			throwables.printStackTrace();
		}
		e.printStackTrace();
	} finally {
		//关闭资源
		JDBCUtils.close(null, preparedStatement, connection);
	}
}

3. 批处理

3.1 基本介绍

在这里插入图片描述

在这里插入图片描述

3.2 应用实例

在这里插入图片描述

特别注意:
1.注意修改配置文件
2.记得关闭事物自动提交(巨坑)

package com.hspedu.jdbc.batch_;

import com.hspedu.jdbc.utils.JDBCUtils;
import org.junit.jupiter.api.Test;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;

/**
* 演示 java 的批处理
*/
public class Batch_ {
	//传统方法,添加 5000 条数据到 admin2
	@Test
	public void noBatch() throws Exception {
		Connection connection = JDBCUtils.getConnection();
		String sql = "insert into admin2 values(null, ?, ?)";
		PreparedStatement preparedStatement = connection.prepareStatement(sql);
		System.out.println("开始执行");
		long start = System.currentTimeMillis();//开始时间
		for (int i = 0; i < 5000; i++) {  //5000 执行
			preparedStatement.setString(1, "jack" + i);
			preparedStatement.setString(2, "666");
			preparedStatement.executeUpdate();
		}
		long end = System.currentTimeMillis();
		System.out.println("传统的方式 耗时=" + (end - start));//传统的方式 耗时=10702
		
		//关闭连接
		JDBCUtils.close(null, preparedStatement, connection);
	}


	//使用批量方式添加数据
	@Test
	public void batch() throws Exception {
		Connection connection = JDBCUtils.getConnection();
		connection.setAutoCommit(false);
		String sql = "insert into admin2 values(null, ?, ?)";
		PreparedStatement preparedStatement = connection.prepareStatement(sql);
		System.out.println("开始执行");
		long start = System.currentTimeMillis();//开始时间
		for (int i = 0; i < 5000; i++) {//5000 执行
			preparedStatement.setString(1, "jack" + i);
			preparedStatement.setString(2, "666");
		//将 sql 语句加入到批处理包中 -> 看源码
		/*
			//1. //第一就创建 ArrayList - elementData => Object[]
			//2. elementData => Object[] 就会存放我们预处理的 sql 语句
			//3. 当 elementData 满后,就按照 1.5 扩容
			//4. 当添加到指定的值后,就 executeBatch
			//5. 批量处理会减少我们发送 sql 语句的网络开销,而且减少编译次数,因此效率提高
			
			public void addBatch() throws SQLException {
				synchronized(this.checkClosed().getConnectionMutex()) {
					if (this.batchedArgs == null) {
						this.batchedArgs = new ArrayList();
					}
					for(int i = 0; i < this.parameterValues.length; ++i) {
						this.checkAllParametersSet(this.parameterValues[i], this.parameterStreams[i], i);
					}
					this.batchedArgs.add(new PreparedStatement.BatchParams(this.parameterValues, this.parameterStreams, this.isStream, this.streamLengths, this.isNull));
				}
			}
		*/
			preparedStatement.addBatch();
		
			//当有 1000 条记录时,在批量执行
			if((i + 1) % 1000 == 0) {//满 1000 条 sql
				preparedStatement.executeBatch();
				//清空一把
				preparedStatement.clearBatch();
			}
		}
		long end = System.currentTimeMillis();
		connection.commit();
		System.out.println("批量方式 耗时=" + (end - start));//批量方式 耗时=108
		//关闭连接
		JDBCUtils.close(null, preparedStatement, connection);
	}
}

4. 数据库连接池

4.1 五k 次连接数据库问题

在这里插入图片描述

package com.hspedu.jdbc.datasource;

import com.hspedu.jdbc.utils.JDBCUtils;
import org.junit.jupiter.api.Test;
import java.sql.Connection;


public class ConQuestion {

	//代码 连接 mysql 5000 次
	@Test
	public void testCon() {
		//看看连接-关闭 connection 会耗用多久
		long start = System.currentTimeMillis();
		System.out.println("开始连接.....");
		
		for (int i = 0; i < 5000; i++) {
			//使用传统的 jdbc 方式,得到连接
			Connection connection = JDBCUtils.getConnection();
			
			//做一些工作,比如得到 PreparedStatement ,发送 sql
			//.......... //关闭
			JDBCUtils.close(null, null, connection);
		}
		long end = System.currentTimeMillis();
		System.out.println("传统方式 5000 次 耗时=" + (end - start));//传统方式 5000 次 耗时=7099
	}
}

4.2 传统获取 Connection 问题分析

在这里插入图片描述

连接示意图
在这里插入图片描述

在这里插入图片描述
连接池示意图
在这里插入图片描述

4.3 数据库连接池种类

在这里插入图片描述

4.3.1 C3P0

继承关系图
在这里插入图片描述

厂家必须实现该接口

在这里插入图片描述
最大连接数和初始化连接数的区别
在这里插入图片描述

前期准备:先导入jar包
在这里插入图片描述

应用实例
在这里插入图片描述
1)方式1

package com.hspedu.jdbc.datasource;

import com.mchange.v2.c3p0.ComboPooledDataSource;
import org.junit.jupiter.api.Test;
import java.io.FileInputStream;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;

/**
* 演示 c3p0 的使用
*/

//方式 1: 相关参数,在程序中指定 user, url , password 等
@Test
public void testC3P0_01() throws Exception {
	//1. 创建一个数据源对象(相当于连接池)
	ComboPooledDataSource comboPooledDataSource = new ComboPooledDataSource();
	
	//2. 通过配置文件 mysql.properties 获取相关连接的信息
	Properties properties = new Properties();
	properties.load(new FileInputStream("src\\mysql.properties"));
	//读取相关的属性值
	String user = properties.getProperty("user");
	String password = properties.getProperty("password");
	String url = properties.getProperty("url");
	String driver = properties.getProperty("driver");
	
	//给数据源 comboPooledDataSource 设置相关的参数
	//注意:连接管理是由 comboPooledDataSource 来管理
	comboPooledDataSource.setDriverClass(driver);
	comboPooledDataSource.setJdbcUrl(url);
	comboPooledDataSource.setUser(user);
	comboPooledDataSource.setPassword(password);
	//设置初始化连接数
	comboPooledDataSource.setInitialPoolSize(10);
	//最大连接数  区别如上图
	comboPooledDataSource.setMaxPoolSize(50);
	
	//测试连接池的效率, 测试对 mysql 5000 次操作
	long start = System.currentTimeMillis();
	for (int i = 0; i < 5000; i++) {
		Connection connection = comboPooledDataSource.getConnection(); //这个方法就是从 DataSource 接口实现的
		//System.out.println("连接 OK");
		connection.close();
	}
	long end = System.currentTimeMillis();
	//c3p0 5000 连接 mysql 耗时=391
	System.out.println("c3p0 5000 连接 mysql 耗时=" + (end - start));
}

2)方式2
需要先添加配置文件:c3p0-config.xml

<c3p0-config>
<!-- 定义名字 -->
  <named-config name="my_c3p0"> 
<!-- 驱动类 -->
  <property name="driverClass">com.mysql.jdbc.Driver</property>
  <!-- url-->
  	<property name="jdbcUrl">jdbc:mysql://127.0.0.1:3306/t1</property>
  <!-- 用户名 -->
  		<property name="user">root</property>
  		<!-- 密码 -->
  	<property name="password">root</property>
  	<!-- 每次增长的连接数-->
    <property name="acquireIncrement">5</property>
    <!-- 初始的连接数 -->
    <property name="initialPoolSize">10</property>
    <!-- 最小连接数 -->
    <property name="minPoolSize">5</property>
   <!-- 最大连接数 -->
    <property name="maxPoolSize">50</property>

	<!-- 可连接的最多的命令对象数 -->
    <property name="maxStatements">5</property> 
    
    <!-- 每个连接对象可连接的最多的命令对象数 -->
    <property name="maxStatementsPerConnection">2</property>
  </named-config>
</c3p0-config>

说明:
在这里插入图片描述

//第二种方式 使用配置文件模板来完成
//1. 将 c3p0 提供的 c3p0.config.xml 拷贝到 src 目录下
//2. 该文件指定了连接数据库和连接池的相关参数

@Test
public void testC3P0_02() throws SQLException {

	ComboPooledDataSource comboPooledDataSource = new ComboPooledDataSource("hsp_edu");
	
	//测试 5000 次连接 mysql
	long start = System.currentTimeMillis();
	
	System.out.println("开始执行....");
	for (int i = 0; i < 500000; i++) {
		Connection connection = comboPooledDataSource.getConnection();
		//System.out.println("连接 OK~");
		connection.close();
	}
	
	long end = System.currentTimeMillis();
	
	//c3p0 的第二种方式 耗时=413
	System.out.println("c3p0 的第二种方式(500000) 耗时=" + (end - start));//1917
}

4.3.2 Druid(德鲁伊)

在这里插入图片描述
先引入jar包和配置文件
druid.properties

#key=value
driverClassName=com.mysql.jdbc.Driver
url=jdbc:mysql://localhost:3306/t1?rewriteBatchedStatements=true
#url=jdbc:mysql://localhost:3306/girls
username=root
password=root
#initial connection Size
initialSize=10
#min idle connecton size  类似最小连接数
minIdle=5
#max active connection size
maxActive=20
#max wait time (5000 mil seconds)  等待最大时长为5秒
maxWait=5000
@Test
public void testDruid() throws Exception {
	//1. 加入 Druid jar 包
	//2. 加入 配置文件 druid.properties , 将该文件拷贝项目的 src 目录
	//3. 创建 Properties 对象, 读取配置文件
	Properties properties = new Properties();
	properties.load(new FileInputStream("src\\druid.properties"));
	
	//4. 创建一个指定参数的数据库连接池, Druid 连接池
	DataSource dataSource = DruidDataSourceFactory.createDataSource(properties);
	long start = System.currentTimeMillis();
	for (int i = 0; i < 500000; i++) {
		Connection connection = dataSource.getConnection();
		System.out.println(connection.getClass());
		//System.out.println("连接成功!");
		connection.close();
	}
	long end = System.currentTimeMillis();
	//druid 连接池 操作 5000 耗时=412
	System.out.println("druid 连接池 操作 500000 耗时=" + (end - start));//539
}

4.4 将 JDBCUtils 工具类改成 Druid(德鲁伊)实现

在这里插入图片描述

/**
* 基于 druid 数据库连接池的工具类
*/
public class JDBCUtilsByDruid {

	private static DataSource ds;
	
	//在静态代码块完成 ds 初始化
	static {
		Properties properties = new Properties();
		try {
			properties.load(new FileInputStream("src\\druid.properties"));
			ds = DruidDataSourceFactory.createDataSource(properties);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	
	//编写 getConnection 方法
	public static Connection getConnection() throws SQLException {
		return ds.getConnection();
	}
	
	//关闭连接, 老师再次强调: 在数据库连接池技术中,close 不是真的断掉连接
	//而是把使用的 Connection 对象放回连接池
	public static void close(ResultSet resultSet, Statement statement, Connection connection) {
		try {
			if (resultSet != null) {
				resultSet.close();
			}
			if (statement != null) {
				statement.close();
			}
			if (connection != null) {
				connection.close();
			}
		} catch (SQLException e) {
			throw new RuntimeException(e);
		}
	}
}

测试类

@SuppressWarnings({"all"})
public class JDBCUtilsByDruid_USE {
	@Test
	public void testSelect() {
	
		System.out.println("使用 druid 方式完成");
		//1. 得到连接
		Connection connection = null;
		
		//2. 组织一个 sql
		String sql = "select * from actor where id >= ?";
		PreparedStatement preparedStatement = null;
		ResultSet set = null;
		
		//3. 创建 PreparedStatement 对象
		try {
			connection = JDBCUtilsByDruid.getConnection();
			System.out.println(connection.getClass());//运行类型 com.alibaba.druid.pool.DruidPooledConnection
			preparedStatement = connection.prepareStatement(sql);
			preparedStatement.setInt(1, 1);//给?号赋值
			
			//执行, 得到结果集
			set = preparedStatement.executeQuery();
			
			//遍历该结果集
			while (set.next()) {
				int id = set.getInt("id");
				String name = set.getString("name");//getName()
				String sex = set.getString("sex");//getSex()
				Date borndate = set.getDate("borndate");
				String phone = set.getString("phone");
				System.out.println(id + "\t" + name + "\t" + sex + "\t" + borndate + "\t" + phone);
			}
		} catch (SQLException e) {
			e.printStackTrace();
		} finally {
			//关闭资源
			JDBCUtilsByDruid.close(set, preparedStatement, connection);
		}
	}
}

5. Apache—DBUtils

5.1 先分析一个问题

在这里插入图片描述

5.2 解决方案

在这里插入图片描述

5.3 用自己的土方法来解决

1)先创建一个类,用于映射

package com.pojo;

/**
 * 该类用于和数据库数据进行绑定
 */
public class Actor {
    private int id;
    private String name;
    private String sex;
    private String borndate;
    private String phone;

    public Actor() {//一定要提供,将来反射要用
    }

    public Actor(int id, String name, String sex, String borndate, String phone) {
        this.id = id;
        this.name = name;
        this.sex = sex;
        this.borndate = borndate;
        this.phone = phone;
    }

    public int getId() {
        return id;
    }

    public String getName() {
        return name;
    }

    public String getSex() {
        return sex;
    }

    public String getBorndate() {
        return borndate;
    }

    public String getPhone() {
        return phone;
    }

    @Override
    public String toString() {
        return "Actor{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", sex='" + sex + '\'' +
                ", borndate='" + borndate + '\'' +
                ", phone='" + phone + '\'' +
                '}';
    }

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

    public void setName(String name) {
        this.name = name;
    }

    public void setSex(String sex) {
        this.sex = sex;
    }

    public void setBorndate(String borndate) {
        this.borndate = borndate;
    }

    public void setPhone(String phone) {
        this.phone = phone;
    }
}

注意事项:
1.必须提供无参构造和set方法,DBUtil框架使用反射时需要使用

2)测试方法

//使用老师的土方法来解决 ResultSet =封装=> Arraylist
@Test
public ArrayList<Actor> testSelectToArrayList() {
	System.out.println("使用 druid 方式完成");
	
	//1. 得到连接
	Connection connection = null;
	//2. 组织一个 sql
	String sql = "select * from actor where id >= ?";
	PreparedStatement preparedStatement = null;
	ResultSet set = null;
	ArrayList<Actor> list = new ArrayList<>();//创建 ArrayList 对象,存放 actor 对象
	
	//3. 创建 PreparedStatement 对象
	try {
		connection = JDBCUtilsByDruid.getConnection();
		System.out.println(connection.getClass());//运行类型 com.alibaba.druid.pool.DruidPooledConnection
		preparedStatement = connection.prepareStatement(sql);
		preparedStatement.setInt(1, 1);//给?号赋值
		
		//执行, 得到结果集
		set = preparedStatement.executeQuery();
		
		//遍历该结果集
		while (set.next()) {
			int id = set.getInt("id");
			String name = set.getString("name");//getName()
			String sex = set.getString("sex");//getSex()
			Date borndate = set.getDate("borndate");
			String phone = set.getString("phone");
			//把得到的 resultset 的记录,封装到 Actor 对象,放入到 list 集合
			list.add(new Actor(id, name, sex, borndate, phone));
		}
		System.out.println("list 集合数据=" + list);
		
		for(Actor actor : list) {
			System.out.println("id=" + actor.getId() + "\t" + actor.getName());
		}
	} catch (SQLException e) {
		e.printStackTrace();
	} finally {
		//关闭资源
		JDBCUtilsByDruid.close(set, preparedStatement, connection);
	}
	//因为 ArrayList 和 connection 没有任何关联,所以该集合可以复用. 
	return list;
}

5.4 DBUtils引入和基本介绍

在这里插入图片描述

5.4.1 应用实例

在这里插入图片描述
入门案例:

//使用 apache-DBUtils 工具类 + druid 完成对表的 crud 操作
@Test
public void testQueryMany() throws SQLException { //返回结果是多行的情况
	//1. 得到 连接 (druid)
	Connection connection = JDBCUtilsByDruid.getConnection();
	
	//2. 使用 DBUtils 类和接口 , 先引入 DBUtils 相关的 jar , 加入到本 Project
	
	//3. 创建 QueryRunner
	QueryRunner queryRunner = new QueryRunner();
	
	//4. 就可以执行相关的方法,返回 ArrayList 结果集
	//String sql = "select * from actor where id >= ?";
	// 注意: sql 语句也可以查询部分列
	String sql = "select id, name from actor where id >= ?";
	
	// 解读
	//(1) query 方法就是执行 sql 语句,得到 resultset ---封装到 --> ArrayList 集合中
	//(2) 返回集合
	//(3) connection: 连接
	//(4) sql : 执行的 sql 语句
	//(5) new BeanListHandler<>(Actor.class): 在将 resultset -> Actor 对象 -> 封装到 ArrayList
	// 底层使用反射机制 去获取 Actor 类的属性,然后进行封装
	//(6) 1 就是给 sql 语句中的? 赋值,可以有多个值,因为是可变参数 Object... params
	//(7) 底层得到的 resultset ,会在 query 关闭, 关闭 PreparedStatment
	/**
	* 分析 queryRunner.query 方法:
	* public <T> T query(Connection conn, String sql, ResultSetHandler<T> rsh, Object... params) throws SQLException {
	* 	PreparedStatement stmt = null;//定义 PreparedStatement
	* 	ResultSet rs = null;//接收返回的 ResultSet
	* 	Object result = null;//返回 ArrayList
	*
	* 	try {
	* 		stmt = this.prepareStatement(conn, sql);//创建 PreparedStatement
	* 		this.fillStatement(stmt, params);//对 sql 进行 ? 赋值
	* 		rs = this.wrap(stmt.executeQuery());//执行 sql,返回 resultset
	* 		result = rsh.handle(rs);//返回的 resultset --> arrayList[result] [使用到反射,对传入 class 对象处理]
	* 	} catch (SQLException var33) {
	* 		this.rethrow(var33, sql, params);
	* 	} finally {
	* 		try {
	* 			this.close(rs);//关闭 resultset
	*   	} finally {
	* 			this.close((Statement)stmt);//关闭 preparedstatement 对象
	* 		}
	* 	}
	* 	return result;
	* }
	*/
	List<Actor> list =
	queryRunner.query(connection, sql, new BeanListHandler<>(Actor.class), 1);
	System.out.println("输出集合的信息");
	for (Actor actor : list) {
	System.out.print(actor);
	}
	//释放资源
	JDBCUtilsByDruid.close(null, null, connection);
}

5.4.2 ResultSetHandler的种类

1)返回的结果是单行记录(单个对象)

//演示 apache-dbutils + druid 
//完成 返回的结果是单行记录(单个对象)
@Test
public void testQuerySingle() throws SQLException {

	//1. 得到 连接 (druid)
	Connection connection = JDBCUtilsByDruid.getConnection();
	//2. 使用 DBUtils 类和接口 , 先引入 DBUtils 相关的 jar , 加入到本 Project
	//3. 创建 QueryRunner
	QueryRunner queryRunner = new QueryRunner();
	//4. 就可以执行相关的方法,返回单个对象
	String sql = "select * from actor where id = ?";
	// 解读
	// 因为我们返回的单行记录<--->单个对象 , 使用的 Hander 是 BeanHandler
	Actor actor = queryRunner.query(connection, sql, new BeanHandler<>(Actor.class), 10);
	System.out.println(actor);
	// 释放资源
	JDBCUtilsByDruid.close(null, null, connection);
}

2)完成查询结果是单行单列-返回的就是 object

//演示 apache-dbutils + druid 
//完成查询结果是单行单列-返回的就是 object
@Test
public void testScalar() throws SQLException {
	//1. 得到 连接 (druid)
	Connection connection = JDBCUtilsByDruid.getConnection();
	//2. 使用 DBUtils 类和接口 , 先引入 DBUtils 相关的 jar , 加入到本 Project
	//3. 创建 QueryRunner
	QueryRunner queryRunner = new QueryRunner();
	//4. 就可以执行相关的方法,返回单行单列 , 返回的就是 Object
	String sql = "select name from actor where id = ?";
	//解读: 因为返回的是一个对象, 使用的 handler 就是 ScalarHandler
	Object obj = queryRunner.query(connection, sql, new ScalarHandler(), 4);
	System.out.println(obj);
	// 释放资源
	JDBCUtilsByDruid.close(null, null, connection);
}

5.4.3 DButils的dml操作

//演示 apache-dbutils + druid 
//完成 dml (update, insert ,delete)
@Test
public void testDML() throws SQLException {
	//1. 得到 连接 (druid)
	Connection connection = JDBCUtilsByDruid.getConnection();
	//2. 使用 DBUtils 类和接口 , 先引入 DBUtils 相关的 jar , 加入到本 Project
	//3. 创建 QueryRunner
	QueryRunner queryRunner = new QueryRunner();
	
	//4. 这里组织 sql 完成 update, insert delete
	//String sql = "update actor set name = ? where id = ?";
	//String sql = "insert into actor values(null, ?, ?, ?, ?)";
	String sql = "delete from actor where id = ?";
	
	//解读
	//(1) 执行 dml 操作是 queryRunner.update()
	//(2) 返回的值是受影响的行数 (affected: 受影响)
	//int affectedRow = queryRunner.update(connection, sql, "林青霞", "女", "1966-10-10", "116");
	int affectedRow = queryRunner.update(connection, sql, 1000 );
	System.out.println(affectedRow > 0 ? "执行成功" : "执行没有影响到表");
	
	// 释放资源
	JDBCUtilsByDruid.close(null, null, connection);
}

5.5 表和 JavaBean 的类型映射关系

在这里插入图片描述

6. DAO 和增删改查通用方法-BasicDao

6.1 问题引入说明

在这里插入图片描述
架构图
在这里插入图片描述

6.2 DAO说明

在这里插入图片描述

6.3 应用实例

在这里插入图片描述
javaBean

/**
* Actor 对象和 actor 表的记录对应
*/
public class Actor { //Javabean, POJO, Domain 对象
	private Integer id;
	private String name;
	private String sex;
	private Date borndate;
	private String phone;
	
	public Actor() { //一定要给一个无参构造器[反射需要]
	}
	
	public Actor(Integer id, String name, String sex, Date borndate, String phone) {
		this.id = id;
		this.name = name;
		this.sex = sex;
		this.borndate = borndate;
		this.phone = phone;
	}
	
	public Integer getId() {
		return id;
	}
	public void setId(Integer id) {
		this.id = id;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public String getSex() {
		return sex;
	}
	public void setSex(String sex) {
		this.sex = sex;
	}
	public Date getBorndate() {
		return borndate;
	}
	public void setBorndate(Date borndate) {
		this.borndate = borndate;
	}
	public String getPhone() {
		return phone;
	}
	public void setPhone(String phone) {
		this.phone = phone;
	}
	@Override
	public String toString() {
		return "\nActor{" +
		"id=" + id +
		", name='" + name + '\'' +
		", sex='" + sex + '\'' +
		", borndate=" + borndate +
		", phone='" + phone + '\'' +
		'}';
	}
}

utils:工具类

/**
* 基于 druid 数据库连接池的工具类
*/
public class JDBCUtilsByDruid {

	private static DataSource ds;
	
	//在静态代码块完成 ds 初始化
	static {
		Properties properties = new Properties();
		try {
			properties.load(new FileInputStream("src\\druid.properties"));
			ds = DruidDataSourceFactory.createDataSource(properties);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	
	//编写 getConnection 方法
	public static Connection getConnection() throws SQLException {
		return ds.getConnection();
	}
	
	//关闭连接, 老师再次强调: 在数据库连接池技术中,close 不是真的断掉连接
	//而是把使用的 Connection 对象放回连接池
	public static void close(ResultSet resultSet, Statement statement, Connection connection) {
		try {
			if (resultSet != null) {
				resultSet.close();
			}
			if (statement != null) {
				statement.close();
			}
			if (connection != null) {
				connection.close();
			}
		} catch (SQLException e) {
			throw new RuntimeException(e);
		}
	}
}

BasicDAO

/**
* 开发 BasicDAO , 是其他 DAO 的父类, 使用到 apache-dbutils
*/
public class BasicDAO<T> { //泛型指定具体类型
	private QueryRunner qr = new QueryRunner();
	
	//开发通用的 dml 方法, 针对任意的表
	public int update(String sql, Object... parameters) {
		Connection connection = null;
		try {
			connection = JDBCUtilsByDruid.getConnection();
			int update = qr.update(connection, sql, parameters);
			return update;
		} catch (SQLException e) {
			throw new RuntimeException(e); //将编译异常->运行异常 ,抛出
		} finally {
			JDBCUtilsByDruid.close(null, null, connection);
		}
	}
	
	//返回多个对象(即查询的结果是多行), 针对任意表
	/**
	*
	* @param sql sql 语句,可以有 ?
	* @param clazz 传入一个类的 Class 对象 比如 Actor.class
	* @param parameters 传入 ? 的具体的值,可以是多个
	* @return 根据 Actor.class 返回对应的 ArrayList 集合
	*/
	public List<T> queryMulti(String sql, Class<T> clazz, Object... parameters) {
		Connection connection = null;
		try {
			connection = JDBCUtilsByDruid.getConnection();
			return qr.query(connection, sql, new BeanListHandler<T>(clazz), parameters);
		} catch (SQLException e) {
			throw new RuntimeException(e); //将编译异常->运行异常 ,抛出
		} finally {
			JDBCUtilsByDruid.close(null, null, connection);
		}
	}


	//查询单行结果 的通用方法
	public T querySingle(String sql, Class<T> clazz, Object... parameters) {
		Connection connection = null;
		try {
			connection = JDBCUtilsByDruid.getConnection();
			return qr.query(connection, sql, new BeanHandler<T>(clazz), parameters);
		} catch (SQLException e) {
			throw new RuntimeException(e); //将编译异常->运行异常 ,抛出
		} finally {
			JDBCUtilsByDruid.close(null, null, connection);
		}
	}
	
	
	//查询单行单列的方法,即返回单值的方法
	public Object queryScalar(String sql, Object... parameters) {
		Connection connection = null;
		try {
			connection = JDBCUtilsByDruid.getConnection();
			return qr.query(connection, sql, new ScalarHandler(), parameters);
		} catch (SQLException e) {
			throw new RuntimeException(e); //将编译异常->运行异常 ,抛出
		} finally {
			JDBCUtilsByDruid.close(null, null, connection);
		}
	}
}

ActorDAO

public class ActorDAO extends BasicDAO<Actor> {
	//1. 就有 BasicDAO 的方法
	//2. 根据业务需求,可以编写特有的方法. 
}

测试类

//测试 ActorDAO 对 actor 表 crud 操作
@Test
public void testActorDAO() {

	ActorDAO actorDAO = new ActorDAO();
	//1. 查询
	List<Actor> actors = actorDAO.queryMulti("select * from actor where id >= ?", Actor.class, 1);
	System.out.println("===查询结果===");
	for (Actor actor : actors) {
		System.out.println(actor);
	}
	
	//2. 查询单行记录
	Actor actor = actorDAO.querySingle("select * from actor where id = ?", Actor.class, 6);
	System.out.println("====查询单行结果====");
	System.out.println(actor);
	
	//3. 查询单行单列
	Object o = actorDAO.queryScalar("select name from actor where id = ?", 6);
	System.out.println("====查询单行单列值===");
	System.out.println(o);
	
	//4. dml 操作 insert ,update, delete
	int update = actorDAO.update("insert into actor values(null, ?, ?, ?, ?)", "张无忌", "男", "2000-11-11", "999");
	System.out.println(update > 0 ? "执行成功" : "执行没有影响表");
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Microsoft Edge】安装详解

文章目录一、下载 Edge1.1 下载网址1.2 版本分类二、安装 Edge2.1 可能的异常情况2.2 安装目录详解2.2.1 Edge 非 Canary 版2.2.2 Edge Canary 版一、下载 Edge Edge 的安装包其实是一个简易安装包&#xff0c;里面封装了一个安装的配置文件&#xff0c;提供真正的安装包下载链…

FreeRTOS 任务基础知识

文章目录一、什么是多任务系统&#xff1f;二、FreeRTOS 任务与协程三、任务状态四、任务优先级五、任务实现六、任务控制块七、任务堆栈RTOS 系统的核心就是任务管理&#xff0c;FreeRTOS 也不例外&#xff0c;而且大多数学习 RTOS 系统的工程师或者学生主要就是为了使用 RTOS…

Revit中怎么绘制多面坡度的屋顶及生成墙

​一、Revit中怎么绘制多面坡度的屋顶 像这种坡屋顶我们可以观察到&#xff0c;它的屋顶轮廓都是带有坡度的&#xff0c;那我可以通过添加定义坡度的方式来绘制出该屋顶。 点击建筑选项卡中的屋顶按钮&#xff0c;选择迹线屋顶。 选择使用拾取线工具&#xff0c;在选项栏中将偏…

从零学习SDK(4)使用SDK创建一个简单的应用程序

SDK&#xff08;Software Development Kit&#xff09;即软件开发工具包&#xff0c;是一组帮助我们开发出软件的工具&#xff0c;包括代码、文档、示例等。一般情况下&#xff0c;我们需要将SDK引入到我们的项目中才能使用它。比如&#xff0c;学Java的朋友最早接触的JDK&…

JMeter使用JDBC Request取样器 获取查询结果

JDBC获取查询结果Java脚本创建文件JSON字符串解析 数据库连接配置定义全局变量JDBC Request 创建文件路径以及文件的脚本 import java.io.FileOutputStream; import java.text.SimpleDateFormat; import java.util.Date; boolean result false; try {//String message new Si…

JAVA初学下(仅做笔记)

一. Map集合&#xff08;双列集合&#xff09; 1.特点 键不能重复&#xff0c;值可以重复 Map接口位于最高层 2.常见API 2.1基本功能 ①注意V put&#xff08;K key,V value&#xff09;这个方法&#xff0c; 当加入 的键值对元素的键(key) 不存在时&#xff0c;就会将 键值…

django项目名称重命名

学会这个小白也能轻松修改项目名称;日常板砖中难免遇到项目多次利用的情况,修改项目名称也成了一门手艺😀;实际操作起来也非常简单,没有那些花花绿绿的东西. 一.项目奔跑 1.拿到项目后,我们先在pycharm运行下,看项目是否能正常奔跑起来; 注意收集正常奔跑后项目的执行环境等…

某程序员哀叹:月薪四五万,却每天极度焦虑痛苦,已有生理性不适,又不敢裸辞,怎么办?...

高薪能买来快乐吗&#xff1f;来看看这位程序员的哀叹&#xff1a;实在是扛不住了&#xff0c;每天都在极度焦虑和痛苦中度过&#xff0c;早上起来要挣扎着做心理建设去上班&#xff0c;已经产生生理性的头晕恶心食欲不振。有工作本身的原因&#xff0c;更多是自己心态的问题&a…

12纳米做出了7纳米的性能,为国产芯片打破美国限制指明道路

美国联合日本、荷兰等限制对中国供应先进芯片设备&#xff0c;试图借此阻止中国发展14纳米以下的先进工艺&#xff0c;然而日前中国一家芯片企业发布了一款全新的芯片&#xff0c;却给中国芯片行业指明了新道路&#xff0c;发展先进性能芯片又了可能性。龙芯近期发布了一款全新…

Flink 优化 (四) --------- 数据倾斜

目录一、判断是否存在数据倾斜二、数据倾斜的解决1. keyBy 后的聚合操作存在数据倾斜2. keyBy 之前发生数据倾斜3. keyBy 后的窗口聚合操作存在数据倾斜一、判断是否存在数据倾斜 相同 Task 的多个 Subtask 中&#xff0c;个别 Subtask 接收到的数据量明显大于其他Subtask 接收…

【DevOps】GitOps之痛 -不完美的GitOps

前言 在前两篇文章中&#xff0c;我们对GitOps进行了大致的介绍&#xff1a; 【DevOps】GitOps初识(上) - 让DevOps变得更好 【DevOps】GitOps初识(下) - 让DevOps变得更好 GitOps 作为软件发布实践方式&#xff0c;有着许多的优点&#xff0c;然而&#xff0c;世上并没有完美…

【Linux】Centos安装mvn命令(maven)

&#x1f341;博主简介 &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01; 文章目录一、下载maven包方法一&#xff1a;官…

CTF流量分析

在CTF里&#xff0c;一些pcapng或pcap文件后缀的数据 不同的数据包有不同的协议&#xff0c;常见的有HTTP&#xff0c;TCP协议 Wireshark 简介 是一个网络封包分析软件。网络封包分析软件的功能是获取网络封包&#xff0c;并尽可能显示出最为详细的网络封包资料 使用WinPC…

舔狗日记:学姐生日快到了,使用Python把她的照片做成视频当礼物

舔狗日记1前言一、需要调入的模块二、实现合并多张图片转成 mp4 视频三、优化改进一下总结前言 这不是学姐生日快到了&#xff0c;于是我学了一手使用Python来把学姐的照片生成为视频&#xff0c;到时候给她一个惊喜&#xff01; 好了先不舔了&#xff0c;下面分享一下用pytho…

基于朴素贝叶斯分类器的钞票真伪识别模型

基于朴素贝叶斯分类器的钞票真伪识别模型 内容 本实验通过实现钞票真伪判别案例来展开学习朴素贝叶斯分类器的原理及应用。 本实验的主要技能点&#xff1a; 1、 朴素贝叶斯分类器模型的构建 2、 模型的评估与预测 3、 分类概率的输出 源码下载 环境 操作系统&#xf…

Leetcode.130 被围绕的区域

题目链接 Leetcode.130 被围绕的区域 mid 题目描述 给你一个 m x n的矩阵 board&#xff0c;由若干字符 X和 O&#xff0c;找到所有被 X围绕的区域&#xff0c;并将这些区域里所有的 O用 X填充。 示例 1&#xff1a; 输入&#xff1a;board [[“X”,“X”,“X”,“X”],[“X…

stm32霸道-lvgl移植学习(一)

文章目录效果有用链接要求创建工程屏幕驱动以及触屏驱动LVGL PortWidgets demo其它效果 目前显示驱动显示较慢&#xff0c;后续会优化。 有用链接 LVGL官网 代码下载 要求 要求最低要求 建议要求架构16、32、64位微控制器或微处理器时钟 > 16 MHz > 48 MHzFlash/RO…

《低代码PaaS驱动集团企业数字化创新白皮书》-平台化加低代码提供破解之道(1)

平台化加低代码提供破解之道 大型企业亟需通过下一代平台开发技术实现软件创新&#xff0c;实现对海量数据的采集加工&#xff0c;以及企业内部数据的互联互通&#xff0c;帮助客户以低成本、短周期、高效率的方式实现数字化应用&#xff0c;进而赋能业务创新。基于此&#xf…

408--计算机网络--网络层总结1

目录 一、网络层概述&#xff1a; 1、网络层的主要任务&#xff1a; 2、网络层向上提供两种服务&#xff1a; 二、IPV4地址分类与子网划分&#xff1a; 1、分类编址&#xff1a; 一、网络层概述&#xff1a; 1、网络层的主要任务&#xff1a; 络层的主要任务就是将分组从…

【数据库基操】启动与连接MySQL数据库

一、启动与关闭 只介绍一种方法&#xff1a; 打开命令行工具&#xff0c;以管理员身份运行 1.启动数据库 net start mysql80 //80是在安装的时候设置的名字&#xff08;默认&#xff09;&#xff0c;不用在意 2.关闭数据库 net stop mysql80 如题已经成功&#…