JDBC(四、连接池和DbUtils)

JDBC(四、连接池和DbUtils)

数据库连接池

为什么使用数据库连接池

不使用数据库连接池

  • 数据库连接资源没有得到很好的重复利用
    • 我们每次使用想数据库建立连接时都要创建一个Connection对象,这个过程要验证IP地址、用户名和密码,用完之后再close断开连接,这样的方式会消耗大量的资源和时间
  • 内存泄漏
    • 数据库连接每次都要创建和关闭,如果程序出现异常为能执行关闭操作,那么就会导致内存泄漏
  • 无法控制连接对象数
    • 如果有几千人需要对数据库进行操作,我们无法控制创建数据库连接对象的数量,大量的资源分配出去,甚至服务器崩溃

数据库连接池的基本思想:

  • 为数据库连接建立一个“缓冲池”,预先在缓冲池中放入一定数量的连接,当需要建立数据库连接时,只需从“缓冲池”取出一个,使用完毕后再放回去
  • 数据库连接池负责分配、管理和释放数据库连接。它允许应用程序重复使用一个现有的数据库连接,而不是重新建立一个。
  • 连接池最大的连接数量,限定了这个连接池能占有的最大连接数,当应用程序向连接池请求的连接数超过最大连接数量时,这些请求将被加入到等待队列中。

使用连接池

  • 资源重复使用
  • 速度快
  • 资源分配良好
  • 统一连接管理,烦那个纸数据库连接泄露

注意:

  • 数据源和数据库连接不同,数据源无需创建多个,它是产生数据库连接的工厂,因此整个应用只需要一个数据源即可
  • 当数据库访问结束后,程序还是可以像以前一样关闭数据库连接,但是这个时候的close()是释放数据库连接,将Connection归还给连接池。
开源的数据库连接池

JDBC的数据库连接池使用javax.sql.DataSource来表示,DataSource只是一个接口,该接口通常由服务器提供实现,一些开源组织提供实现:

  • DBCP:Apache提供的数据库连接池,速度比C3P0快
  • C3P0:开源组织提供的数据库连接池,速度一般,较稳定
  • Proxool:SourceForge下的一个开源项目的数据库连接池,有监控连接池的功能
  • BoneCP:开源组织提供的数据库连接池,速度快
  • Druid:阿里提供的数据库连接池,据说是及DBCP、C3P0、Proxool优点于一身的数据库连接池
Druid

阿里巴巴开源平台上的一个数据库连接池实现,加入了日志监控,可以很好的监控DB池连接和SQL的执行情况,可以说是针对监控而生的DB连接池,据说是目前最好的连接池。

连接池的使用:

  • 导入jar包

  • 准备properties配置文件(我放在了src目录下)

    • 数据库连接池中有很多的配置参数,自行参考官网
    # 数据库用户名
    username=root
    # 数据库密码
    password=root
    # 数据库连接url
    url=jdbc:mysql://localhost:3306/my_db01?useUnicode=true&characterEncoding=utf8&serverTimezone=GMT%2B8
    # 数据库驱动类位置
    driverClassName=com.mysql.cj.jdbc.Driver
    ## 初始数量
    initialSize=5
    ## 最大数量
    maxActive=10
    ## 最小数量
    minIdle=3
    
  • 使用数据库连接池获取连接对象

        public static void main(String[] args) {
            //获取配置druid.properties文件
            InputStream is = Thread.currentThread().getContextClassLoader().getResourceAsStream("druid.properties");
            Properties p = new Properties();
            try {
                p.load(is);
            } catch (IOException e) {
                e.printStackTrace();
            }
            try {
                //创建数据库连接池对象
                DataSource dataSource = DruidDataSourceFactory.createDataSource(p);
                //在从连接池中获取一个Connection对象
                Connection conn = dataSource.getConnection();
                System.out.println(conn);
                conn.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    
ThreadLocal类

代表一个线程的局部变量。

JDK1.2就提供了java.lang.ThreadLocal,为解决多线程程序并发问题提供了一种新的思路。

ThreadLocal用于保存一个共享变量的副本。

我们这里使用它仅为解决一个问题:

  • 当一个线程在执行事务时,每执行一个DAO操作就会从连接池中获得一次Connection,但是一个事务中会有多个DAO操作,那么就不能保证这个线程在每次执行DAO操作时,使用的是同一个Connection对象,如果每一次执行不是同一个Connection对象,根本就无法完成一个事务

  • 所以,我们使用ThreadLocal,它根据从连接池中获取的Connection提供一个Connection的副本,当我们这个线程需要使用Connection时,就可以使用这个线程中ThreadLocal提供的副本Connection对象,这个Connection只属于这个线程,当这个线程使用副本的Connection执行完事务后,将其关闭掉就可以了

封装JDBCTools
package com.bdit.util;

import com.alibaba.druid.pool.DruidDataSourceFactory;

import javax.sql.DataSource;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Properties;

public class JdbcTools {
   //准备连接池
   private static DataSource dataSource;
   private static ThreadLocal<Connection> threadLocal;
   static {
       Properties properties = new Properties();
       try {
           properties.load(Thread.currentThread().getContextClassLoader().getResourceAsStream("druid.properties"));
           dataSource = DruidDataSourceFactory.createDataSource(properties);
           threadLocal = new ThreadLocal<>();
       } catch (Exception e) {
           e.printStackTrace();
       }
   }

   public static Connection getConnection(){
       Connection connection = threadLocal.get();
       if (connection==null){
           try {
               connection = dataSource.getConnection();
               threadLocal.set(connection);
           } catch (SQLException throwables) {
               throwables.printStackTrace();
           }
       }
       return connection;
   }

   public static void releaseConnection(){
       Connection connection = threadLocal.get();
       if (connection!=null){
           try {
               connection.close();
               threadLocal.remove();
           } catch (SQLException throwables) {
               throwables.printStackTrace();
           }
       }
   }
}

Apache-DbUtils

Apache组织提供的一个开源的JDBC工具类库,它是对JDBC的简单封装,学习成本极低,并且使用DbUtils能极大简化jdbc编码的工作量,同时也不会影响程序的性能

DbUtils类

提供了如何关闭连接、装载JDBC驱动程序等常规工作的工具类,里面的方法都是静态的

常用方法:

返回值方法名说明
static voidclose(···)提供了三个重载的关闭方法(Connection、 Statement、ResultSet),这些方法检查所提供的参数是不是NULL,如果不是就关闭它们
static voidcloseQuietly(···)这类方法不仅能在Connection、Statement、ResultSet为空时避免关闭,还能隐藏一些在程序中抛出的SQLException
static voidcommitAndClose(Connection conn)提交事务,并关闭连接
static voidcommitAndCloseQuietly(Connection conn)交事务,并关闭连接,而且不会跑出SQL异常
static voidrollback(Connection conn)事务回滚,允许conn为空,因为在方法的内部做了判断
static voidrollbackAndClose(Connection conn)事务回滚,并关闭连接
static voidrollbackAndCloseQuietly(Connection conn)事务回滚,并关闭连接,而且不会跑出SQL异常
static booleanloadDriver(String driverClassName)装载并注册JDBC驱动程序,成功返回true,使用这个方法就不需要处理ClassNotFoundException
QueryRunner类

封装了SQL的执行,是线程安全的

  • 可以完成增、删、改、查、批处理
  • 而且考虑到了一个事务的处理需要共用Connection
  • 该类最主要的还是简化了SQL查询,它与ResultSetHandler组合在一起使用可以完成大部分的数据库操作,极大的减少了代码量

构造方法:

  • QueryRunner();默认的构造方法
  • QueryRunner(DataSource ds);需要一个javax.sql.DataSource作为参数

常用方法:

返回值方法名说明
intupdate(Connection conn,String sql,Object… params)执行一个跟更新操作(插入、修改、删除),返回的是影响的行数
Tinsert(Connection conn,String sql,ResultSetHandler rsh,Object… params)执行一个插入操作,只支持insert
int[]baech(Connection conn,String sql,Object[] params)批处理,支持insert、update、delete
TinsertBatch(Connection conn,String sql,ResultSetHandler rsh,Object[] params)批处理,只支持insert语句
Objectquery(Connection conn,String sql,ResultSetHandler rsh,Object… params)执行一个查询操作,
ResultSetHandler接口

用于处理java.sql.ResultSet,将数据按要求转换为另一种形式。

此接口提供了一个单的的方法将返回值作为QueryRunner类的query()方法的返回值

该接口中有如下实现类可以使用:

  • ArrayHandler:把结果集中的第一行数据转换成对象数值
  • ArrayListHandler:把结果集中的每一行数据都转换成一个数组,然后再放入List中
  • BeanHandler:将结果集中的第一行数据封装到一个对应的JavaBean实例中
  • BranListHandler:将结果集中的每一行数据都封装到一个对应的JavaBean实例中,然后再放入List中
  • ColumnListHandler:将结果集中某一列的数据存放到List中
  • KeyedHandler:将结果集中的每一行数据都封装到一个Map中,再把这些map再存到一个map中,其key为指定的key
  • MapHandler:将结果集中的第一行数据封装到一个Map中,key是列名,value就是对应值
  • MapListHandler:将结果集中的每一行数据都封装到一个Map中,然后再放入List中
表与JavaBean

实际开发中对于数据库中的表数据,都会有与之对应的JavaBean,

  • 在自定义JavaBean时,基本数据类型的变量都使用其包装类,因为数据库中的所有类型都有可能为NULL
  • 如果在一些特殊情况下,JavaBean的实体类的属性名无法和数据库的列名相同,那么在执行SQL语句时,可以使用起别名的方式与之对应
示例:

(使用的是上面封装的JDBCTools,Account自定义的JavaBean和数据库中列相匹配)

package com.bdit;

import com.bdit.bean.Account;
import com.bdit.util.JdbcTools;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import org.apache.commons.dbutils.handlers.BeanListHandler;
import org.apache.commons.dbutils.handlers.ScalarHandler;

import java.sql.Connection;
import java.sql.SQLException;
import java.util.List;

/**
 * commons-DbUtils
 */
public class JdbcTest3 {
    private static QueryRunner queryRunner= new QueryRunner();

    public static void main(String[] args) {
        Account account = new Account();
        account.setName("xiaoming");
        account.setMoney(9999);
//        insert(account);
//        delete(2004);
//        update(4,account);
//        findById(1);
//        findAll();
        findCount();
    }

    public static void insert(Account account){
        String sql = "insert into account(name,money) values(?,?)";
        Object[] params = {account.getName(),account.getMoney()};
        try(Connection connection = JdbcTools.getConnection()){
            int n = queryRunner.update(connection,sql,params);
            if (n>0){
                System.out.println("-----------添加成功----------");
            }else {
                System.out.println("-----------添加失败----------");
            }
        } catch (SQLException throwables) {
            throwables.printStackTrace();
        }
    }
    public static void delete(int id){
        String sql = "delete from account where id = ?";
        try(Connection connection = JdbcTools.getConnection()) {
            int i  =queryRunner.update(connection,sql,id);
            if (i>0){
                System.out.println("-----------删除成功----------");
            }else {
                System.out.println("-----------删除失败----------");
            }
        } catch (SQLException throwables) {
            throwables.printStackTrace();
        }
    }

    public static void update(int id,Account account){
        String sql = "update account set name=?,money=? where id=?";
        Object[] params = {account.getName(),account.getMoney(),id};
        try(Connection connection = JdbcTools.getConnection()){
            int i  = queryRunner.update(connection,sql,params);
            if (i>0){
                System.out.println("-----------修改成功----------");
            }else {
                System.out.println("-----------修改失败----------");
            }
        }catch (SQLException throwables) {
            throwables.printStackTrace();
        }
    }

    public static void findById(int id){
        String sql ="select * from account where id =?";
        try(Connection connection  = JdbcTools.getConnection()) {
            Account account = queryRunner.query(connection,sql,new BeanHandler<Account>(Account.class),id);
            System.out.println(account);
        } catch (SQLException throwables) {
            throwables.printStackTrace();
        }
    }
    public static void findAll(){
        String sql = "select * from account";
        try(Connection connection = JdbcTools.getConnection()) {
            List<Account> list = queryRunner.query(connection,sql,new BeanListHandler<Account>(Account.class));
            for (Account a : list){
                System.out.println(a);
            }
        } catch (SQLException throwables) {
            throwables.printStackTrace();
        }
    }

    public static void findCount(){
        String sql ="select count(id) from account";
        try(Connection connection = JdbcTools.getConnection()) {
            Long num = queryRunner.query(connection,sql,new ScalarHandler<Long>());
            System.out.println("表中共"+num+"行");
        } catch (SQLException throwables) {
            throwables.printStackTrace();
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值