我在GlassFish上有一个Java-JSF Web应用程序,我想在其中使用连接池。因此,我创建了一个application
范围的bean,为其他bean的Connection
实例提供服务:
public class DatabaseBean {
private DataSource myDataSource;
public DatabaseBean() {
try {
Context ctx = new InitialContext();
ecwinsDataSource = (DataSource) ctx.lookup("jdbc/myDataSource");
} catch (NamingException ex) {
ex.printStackTrace();
}
}
public Connection getConnection() throws ClassNotFoundException, SQLException, InstantiationException, IllegalAccessException {
Connection connection = myDataSource.getConnection();
System.out.println("Succesfully connected: " + connection);
//Sample: Succesfully connected: com.sun.gjc.spi.jdbc40.ConnectionHolder40@7fb213a5
return connection;
}
}
这样连接池的填充速度非常快;在通过“db-related”视图进行一些导航后,应用程序将停止并显示以下内容:
RAR5117:无法从连接池[mysql_testPool]获取/创建连接。原因:正在使用的连接等于max-pool-size和expired max-wait-time。无法分配更多连接。 RAR5114:分配连接时出错:[分配连接时出错。原因:正在使用的连接等于max-pool-size和expired max-wait-time。无法分配更多连接。] java.sql.SQLException:分配连接时出错。原因:正在使用的连接等于max-pool-size和expired max-wait-time。无法分配更多连接。
我正在关闭每种方法中的连接和其他资源。应用程序通过独立连接运行一切正常。
我做错了什么?任何提示或建议将不胜感激。
答案 0 :(得分:20)
该异常表示泄漏数据库连接的应用程序代码的典型情况。您需要确保获得和关闭所有{(1}},Connection
和 Statement
){{3根据普通的JDBC习惯用法阻塞在同一个方法块中。
ResultSet
或者当你不在Java 7上时,在public void create(Entity entity) throws SQLException {
try (
Connection connection = dataSource.getConnection();
PreparedStatement statement = connection.prepareStatement(SQL_CREATE);
) {
statement.setSomeObject(1, entity.getSomeProperty());
// ...
statement.executeUpdate();
}
}
块中。在try-finally
中关闭它们将保证在例外情况下它们也会关闭。
finally
是的,即使使用连接池,您仍需要自己关闭连接。在初学者中,他们认为它会自动处理结束是一个常见的错误。这不正确。连接池返回一个包装连接,它在close()中执行类似下面的操作:
public void create(Entity entity) throws SQLException {
Connection connection = null;
PreparedStatement statement = null;
try {
connection = dataSource.getConnection();
statement = connection.prepareStatement(SQL_CREATE);
statement.setSomeObject(1, entity.getSomeProperty());
// ...
statement.executeUpdate();
} finally {
if (statement != null) try { statement.close(); } catch (SQLException logOrIgnore) {}
if (connection != null) try { connection.close(); } catch (SQLException logOrIgnore) {}
}
}
不关闭它们会导致连接没有被释放回池中以便重复使用,因此它将一次又一次地获取一个新连接,直到数据库用尽连接,这将导致应用程序崩溃。
答案 1 :(得分:0)
如果您需要JDBC连接池,为什么不依赖已有的连接池? AFAIK,JDBC连接池在这些Java应用程序服务器中被认为或多或少是标准功能,而IMO,如果您只是对创建应用程序感兴趣,则不应该自己构建它。
这是一个可以帮助您入门的链接: http://weblogs.java.net/blog/2007/09/12/totd-9-using-jdbc-connection-pooljndi-name-glassfish-rails-application
你应该做的是找出如何让你的应用程序使用jndi从池中获取连接。