package com.xiaoleilu.hutool.db.ds.c3p0; import java.util.Collection; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; import javax.sql.DataSource; import com.mchange.v2.c3p0.ComboPooledDataSource; import com.xiaoleilu.hutool.db.DbRuntimeException; import com.xiaoleilu.hutool.db.DbUtil; import com.xiaoleilu.hutool.db.ds.DSFactory; import com.xiaoleilu.hutool.setting.Setting; import com.xiaoleilu.hutool.util.CollectionUtil; import com.xiaoleilu.hutool.util.StrUtil; /** * Druid数据源工厂类 * @author Looly * */ public class C3p0DSFactory extends DSFactory { public static final String DS_NAME = "C3P0"; /** 数据源池 */ private Map<String, ComboPooledDataSource> dsMap; public C3p0DSFactory() { this(null); } public C3p0DSFactory(Setting setting) { super(DS_NAME, setting); checkCPExist(ComboPooledDataSource.class); this.dsMap = new ConcurrentHashMap<>(); } @Override synchronized public DataSource getDataSource(String group) { if (group == null) { group = StrUtil.EMPTY; } // 如果已经存在已有数据源(连接池)直接返回 final ComboPooledDataSource existedDataSource = dsMap.get(group); if (existedDataSource != null) { return existedDataSource; } ComboPooledDataSource ds = createDataSource(group); // 添加到数据源池中,以备下次使用 dsMap.put(group, ds); return ds; } @Override public void close(String group) { if (group == null) { group = StrUtil.EMPTY; } ComboPooledDataSource ds = dsMap.get(group); if (ds != null) { ds.close(); dsMap.remove(group); } } @Override public void destroy() { if(CollectionUtil.isNotEmpty(dsMap)){ Collection<ComboPooledDataSource> values = dsMap.values(); for (ComboPooledDataSource ds : values) { ds.close(); } dsMap.clear(); } } /** * 创建数据源 * @param group 分组 * @return C3P0数据源 {@link ComboPooledDataSource} */ private ComboPooledDataSource createDataSource(String group){ final Setting config = setting.getSetting(group); if(CollectionUtil.isEmpty(config)){ throw new DbRuntimeException("No C3P0 config for group: [{}]", group); } final ComboPooledDataSource ds = new ComboPooledDataSource(); //基本信息 ds.setJdbcUrl(getAndRemoveProperty(config, "url", "jdbcUrl")); ds.setUser(getAndRemoveProperty(config, "username", "user")); ds.setPassword(getAndRemoveProperty(config, "password", "pass")); final String driver = getAndRemoveProperty(config, "driver", "driverClassName"); try { if(StrUtil.isNotBlank(driver)){ ds.setDriverClass(driver); }else{ ds.setDriverClass(DbUtil.identifyDriver(ds.getJdbcUrl())); } } catch (Exception e) { throw new DbRuntimeException(e); } config.toBean(ds);//注入属性 return ds; } /** * 获得指定KEY对应的值,key1和key2为属性的两个名字,可以互作别名 * @param properties 属性 * @param key1 属性名 * @param key2 备用属性名 * @return 值 */ private String getAndRemoveProperty(Setting setting, String key1, String key2){ String value = (String) setting.remove(key1); if(StrUtil.isBlank(value)){ value = (String) setting.remove(key2); } return value; } }