写了个通用连接池,c3p0,用来连接多个数据库,大家看看有没有问题

miracleliu 2013-11-26 05:00:06
要实现的功能,就是读取jdbc.properties,获取数据库连接。
只要在jdbc.properties配置了url,username,password三项就可以调用DBPool实现数据库连接,是连接池的。


import java.beans.PropertyVetoException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;

import org.apache.log4j.Logger;

import com.mchange.v2.c3p0.ComboPooledDataSource;

public class DBPool {

private static Logger log = Logger.getLogger(DBPool.class);
private static DBPool dbPool = new DBPool();

private static Map<String, ComboPooledDataSource> dataSources = new ConcurrentHashMap<String, ComboPooledDataSource>();

private DBPool() {
}

public final static DBPool getInstance() {
return dbPool;
}

public final Connection getConnection(String connName) {
try {
Connection conn = getDataSource(connName).getConnection();
return conn;
} catch (SQLException e) {
log.error("can't get the connection :" + e);
throw new RuntimeException("unable to connect to the database ", e);
}
}

private ComboPooledDataSource getDataSource(String connName) {

ComboPooledDataSource dataSource = dataSources.get(connName);

if (dataSource != null) {
return dataSource;
}

try {
InputStream is = DBPool.class.getResourceAsStream("/jdbc.properties");
Properties p = new Properties();
p.load(is);
String dbUrl = p.getProperty(connName + ".url");
String dbUser = p.getProperty(connName + ".username");
String dbPwd = p.getProperty(connName + ".password");
String driver = p.getProperty("driver");
dataSource = new ComboPooledDataSource();
dataSource.setUser(dbUser);
dataSource.setPassword(dbPwd);
dataSource.setJdbcUrl(dbUrl);
dataSource.setDriverClass(driver);
dataSource.setInitialPoolSize(2);
dataSource.setMinPoolSize(1);
dataSource.setMaxPoolSize(10);
dataSource.setMaxStatements(50);
dataSource.setMaxIdleTime(60);
dataSource.setAcquireIncrement(5);
is.close();
} catch (PropertyVetoException e) {
log.error("jdbc.properties error ", e);
} catch (Exception e) {
log.error("datasource generate error ", e);
}

dataSources.put(connName, dataSource);

return dataSource;
}

public final void close(Statement pstmt, ResultSet rs, Connection conn) {
try {
if (pstmt != null)
pstmt.close();
if (rs != null)
rs.close();
if (conn != null)
conn.close();
} catch (SQLException e) {
log.error("close conn error ", e);
}
}


}

写了一个测试代码,执行正确

public class Test {
public static void main(String[] args) {
Connection conn = DBPool.getInstance().getConnection("erpt");
PreparedStatement pstmt = null;
ResultSet rs = null;
try{
pstmt= conn.prepareStatement("select * from daily_aop limit 5;");
rs = pstmt.executeQuery();
while (rs.next()) {
System.out.println(rs.getInt(1));
}
} catch (SQLException e) {
e.printStackTrace();
} finally {
DBPool.getInstance().close(pstmt, rs, conn);
}

}

}



...全文
763 16 打赏 收藏 转发到动态 举报
写回复
用AI写文章
16 条回复
切换为时间正序
请发表友善的回复…
发表回复
qinghechaoge 2015-12-20
  • 打赏
  • 举报
回复
引用 15 楼 miraclestar 的回复:
[quote=引用 14 楼 kky2010_110 的回复:] 我看不出你这个关闭是放回连接池中,你的conn是sql包中的connection所以你直接关掉了。
看这个火龙果大大的回复:http://bbs.csdn.net/topics/330195313 c3p0是要关闭的。其他的连接池不了解 [/quote] 楼主虽然先的是conn.close() 但这也不是关闭了连接,而是将连接放回了连接池
miracleliu 2013-12-13
  • 打赏
  • 举报
回复
引用 14 楼 kky2010_110 的回复:
我看不出你这个关闭是放回连接池中,你的conn是sql包中的connection所以你直接关掉了。
看这个火龙果大大的回复:http://bbs.csdn.net/topics/330195313 c3p0是要关闭的。其他的连接池不了解
kky2010_110 2013-12-13
  • 打赏
  • 举报
回复
我看不出你这个关闭是放回连接池中,你的conn是sql包中的connection所以你直接关掉了。
kky2010_110 2013-12-13
  • 打赏
  • 举报
回复
我看不出你这个关闭是放回连接池中你的conn是sql包中的connection所以你直接关掉了。
cccooddeee 2013-12-12
  • 打赏
  • 举报
回复
MARK 正想把部分业务处理使用另外一个库去执行。
miracleliu 2013-11-28
  • 打赏
  • 举报
回复
引用 8 楼 bao110908 的回复:
虽然使用了 ConcurrentHashMap 但并未处理 ComboPooledDataSource dataSource = dataSources.get(connName); if (dataSource != null) { return dataSource; } 这块代码的同步问题,如果竞争激烈的话,很可能会造成连接泄漏。
这里不会吧,ConcurrentHashMap只是读取数据,put的时候可能会有点问题, put的时候加个锁就行了
synchronized (dataSources) {
			dataSources.put(connName, dataSource);
		}
引用 6 楼 scottxzj 的回复:
直接配置多数据源就可以了
怎么配置?
miracleliu 2013-11-28
  • 打赏
  • 举报
回复
引用 9 楼 bao110908 的回复:
另外,对于异常处理非常不好,比如: } catch (PropertyVetoException e) { log.error("jdbc.properties error ", e); } catch (Exception e) { log.error("datasource generate error ", e); } dataSources.put(connName, dataSource); return dataSource; 直接把异常吃了,如果 dataSource 没有初始化成功,也将 put 到 dataSources 中,并返回给调用者一个 null 值的 dataSource。
谢谢火龙果! 那么这个异常怎么处理比较好呢?throw 一个运行是异常吗?
  • 打赏
  • 举报
回复
另外,对于异常处理非常不好,比如: } catch (PropertyVetoException e) { log.error("jdbc.properties error ", e); } catch (Exception e) { log.error("datasource generate error ", e); } dataSources.put(connName, dataSource); return dataSource; 直接把异常吃了,如果 dataSource 没有初始化成功,也将 put 到 dataSources 中,并返回给调用者一个 null 值的 dataSource。
  • 打赏
  • 举报
回复
虽然使用了 ConcurrentHashMap 但并未处理 ComboPooledDataSource dataSource = dataSources.get(connName); if (dataSource != null) { return dataSource; } 这块代码的同步问题,如果竞争激烈的话,很可能会造成连接泄漏。
snow-is-my-Love 2013-11-27
  • 打赏
  • 举报
回复
直接配置多数据源就可以了
末日哥 2013-11-26
  • 打赏
  • 举报
回复
额,你这里调用的close应该是放回连接池并不是关闭物理连接的,之前我没想对
miracleliu 2013-11-26
  • 打赏
  • 举报
回复
引用 3 楼 fudongrifdr 的回复:
木有认真看,还以为你是自己实现了一个连接池。。 你这close是把连接用完以后关闭了么,不是还给连接池?
c3p0连接池,要直接关掉的。 http://bbs.csdn.net/topics/390458524?page=1#post-394497078
末日哥 2013-11-26
  • 打赏
  • 举报
回复
木有认真看,还以为你是自己实现了一个连接池。。 你这close是把连接用完以后关闭了么,不是还给连接池?
miracleliu 2013-11-26
  • 打赏
  • 举报
回复
引用 1 楼 songbgi 的回复:
你这写的是链接池?
准确的说不是连接池实现,是连接池的使用,哈哈。 主要功能就是可以调用多个数据库连接池。只要配置文件里写一下就可以在代码里调用了。
  • 打赏
  • 举报
回复
你这写的是链接池?

67,513

社区成员

发帖
与我相关
我的任务
社区描述
J2EE只是Java企业应用。我们需要一个跨J2SE/WEB/EJB的微容器,保护我们的业务核心组件(中间件),以延续它的生命力,而不是依赖J2SE/J2EE版本。
社区管理员
  • Java EE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧