java - 没有框架的 HikariCP

标签 java mysql hikaricp

我是第一次在 Java 中使用 HikariCP,但没有使用任何后端框架或 JPA。

我的项目有以下包结构

----db
----controllers
----views
----models
----handlers

db.DBManager.java 中设置了 hikari 数据源:

public class DBManager {
final protected HikariDataSource hikariDS;
protected Connection connection = null;

public DBManager() {

    String resourceName = "re/iprocu/coperativeerp/config/db/dbSettings.properties"; // could also be a constant
    ClassLoader loader = Thread.currentThread().getContextClassLoader();
    Properties props = new Properties();
    try(InputStream resourceStream = loader.getResourceAsStream(resourceName)) {
        props.load(resourceStream);
    } catch (IOException ex) {
        Logger.getLogger(DBManager.class.getName()).log(Level.SEVERE, null, ex);
    }
    final HikariConfig hikariConfig = new  HikariConfig(props);
    //hikariConfig.setConnectionInitSql(absoluteFilePath);
    hikariConfig.setMaximumPoolSize(30);
    hikariConfig.setMinimumIdle(10);
    hikariConfig.setPoolName("TripPool");
    hikariConfig.setConnectionTimeout(1000);
    hikariConfig.setConnectionTestQuery("SELECT 1");
    hikariConfig.setAutoCommit(true);
    hikariConfig.setInitializationFailFast(false);
    hikariConfig.setLeakDetectionThreshold(2000);
    this.hikariDS = new HikariDataSource(hikariConfig);
}
}

handlers 包中是放置对数据库的查询的地方。 处理程序类扩展 DBManager 以获取初始化的 hikari 数据源。

package re.iprocu.coperativeerp.handlers;
public class InventoryHandler extends DBManager{

   final static private Logger logger = Logger.getLogger(SupplierHandler.class.getName());

public InventoryHandler(){
    super();
} 


public int addProductToInventory(Inventory inv){
    int key = 0;

    PreparedStatement ps = null;
    try{
        connection  = hikariDS.getConnection();
        ps = connection.prepareStatement(sql);

        ps.executeUpdate();

        generatedKeys = ps.getGeneratedKeys();
        if (generatedKeys != null && generatedKeys.next()) {
            key = generatedKeys.getInt(1);
        } else {
            System.out.println("No Key");
        }
    }catch(SQLException e){ e.printStackTrace();}
    finally{
        if( connection != null ){
            try {connection.close();} 
            catch (SQLException ex) {logger.log(Level.SEVERE, null, ex);}
        }
        if( ps != null){
            try { ps.close();} 
            catch (SQLException ex) { logger.log(Level.SEVERE, null, ex);}
        }
    }

    return key;
}
}

controllers 包中,我定义并初始化了我需要使用的处理程序,并调用相应的方法。

问题是在刷新 ui 或有时第一次加载后,抛出错误 java.sql.SQLTransientConnectionException: TripPool - 连接不可用,请求在 1001 毫秒后超时。

Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: 数据源拒绝建立连接,来自服务器的消息:“太多连接”

根据我的项目结构,配置 HikariCP 并最大限度地利用它的最佳方式是什么

编辑:

我发现这个 stackoverflow Question它对我有用,但我担心的是解决方案有一个 Double-checked locking 问题,在讨论中没有解决,我不知道该怎么做

最佳答案

您的错误消息说,您的数据库当前打开的连接太多。所以首先你可以查阅你的数据库文档并增加连接限制。或者与您的数据库管理员交谈(如果有的话)。

可能的原因是您为 DBManager 的任何子类的每个新实例创建了新的数据源。使您的 hikariDS 成为静态字段并在静态 block 中对其进行初始化:

public class DBManager {
  public static final HikariDataSource hikariDS;

  static {
    String resourceName = "re/iprocu/coperativeerp/config/db/dbSettings.properties";
    ClassLoader loader = Thread.currentThread().getContextClassLoader();
    Properties props = new Properties();
    try(InputStream resourceStream = loader.getResourceAsStream(resourceName)) {
      props.load(resourceStream);
    } catch (IOException ex) {
      Logger.getLogger(DBManager.class.getName()).log(Level.SEVERE, null, ex);
    }

    final HikariConfig hikariConfig = new  HikariConfig(props);
    hikariConfig.setMaximumPoolSize(30);
    hikariConfig.setMinimumIdle(10);
    hikariConfig.setPoolName("TripPool");
    hikariConfig.setConnectionTimeout(1000);
    hikariConfig.setConnectionTestQuery("SELECT 1");
    hikariConfig.setAutoCommit(true);
    hikariConfig.setInitializationFailFast(false);
    hikariConfig.setLeakDetectionThreshold(2000);
    DBManager.hikariDS = new HikariDataSource(hikariConfig);
  }
}

我还删除了 connection 字段。您最好将其存储在方法的局部变量中。

关于java - 没有框架的 HikariCP,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38302025/

相关文章:

java - 使用变量作为文件路径将 ImageIcon 添加到 JPanel

java - 环境变量 DHIS2_HOME : null (TrayApp. java [main])

java - 执行 Java 代码将 DB 表数据导出到 Excel 工作表 (.xlsx) 时内存不足

java - ERROR : java. sql.SQLSyntaxErrorException:您的 SQL 语法有错误;

java - 这是通过 jdbc 连接和断开 mysql 的安全方法吗?

hikaricp - Hikari CP maxLifetime/idleTimeout

java - 编程技巧: not able to modify variable before constructor call

mysql - SQL OR NOT EXISTS 不起作用

java - JDBC连接是如何实现的?

Java JDBC mysql 连接池