注:有些java类是JDK没有的,请到 http://www.findjar.com/index.x 中查找对应的jar包
1. 数据库配置文件 (放在工程的 classpath 下)
database.properties
## oracle ## #c3p0.driver=oracle.jdbc.driver.OracleDriver #c3p0.url=jdbc:oracle:thin:@192.168.1.49:1523:DB02 ## DB2 ## c3p0.driver=com.ibm.db2.jcc.DB2Driver c3p0.url=jdbc:db2://172.20.15.174:50000/test c3p0.user=admin c3p0.password=admin c3p0.initialPoolSize=15 #连接池中保留的最大连接数 c3p0.maxPoolSize=20 #连接池中保留的最小连接数 c3p0.minPoolSize=10 #最大空闲时间,n秒内未使用则连接被丢弃。若为0则永不丢弃 c3p0.maxIdleTime=60 #获得连接的最大等待毫秒数(两次连接中间隔时间) c3p0.acquireRetryDelay=1000 #当连接池中的连接耗尽的时候c3p0一次同时获取的连接数 c3p0.acquireIncrement=15 #定义在从数据库获取新连接失败后重复尝试的次数 c3p0.acquireRetryAttempts=30 #连接关闭时默认将所有未提交的操作回滚 c3p0.autoCommitOnClose=false #连接池内单个连接所拥有的最大缓存statements数 c3p0.maxStatementsPerConnection=100 #通过多线程实现多个操作同时被执行 c3p0.numHelperThreads=10 ### DPCP ### #最大等待秒数,单位为毫秒, 超过时间会报出错误信息 dpcp.maxwait=60000
2. 配置文件解析类
import java.io.IOException; import java.io.InputStream; import java.util.Iterator; import java.util.Map; import java.util.Map.Entry; import java.util.Properties; import java.util.Set; import org.apache.log4j.Logger; /** * @ClassName: ParsePropertiesFile * @Description: 解析属性文件(*.properties) * @author * @company * @date 2012-11-8 * @version V1.0 */ public final class ParsePropertiesFile { private static final Logger LOG = Logger .getLogger(ParsePropertiesFile.class); private static final ParsePropertiesFile instance = new ParsePropertiesFile(); private ParsePropertiesFile() { } public static ParsePropertiesFile getInstance() { return instance; } /** * @Title: getConfig * @Description: 解析并读取属性文件(*.properties) * @param propFile * 属性文件路径 * @return:Map * @author * @date 2012-11-8 */ public MapgetConfig(String propFile) { InputStream is = null; try { is = this.getClass().getClassLoader().getResourceAsStream(propFile); Properties prop = new Properties(); prop.load(is); Set > set = prop.entrySet(); Iterator > it = set.iterator(); String key = null, value = null; Entry
3. IOUtil
import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.Reader; import org.apache.log4j.Logger; public final class IOUtil { private static final Logger LOG = Logger.getLogger(IOUtil.class); private IOUtil() { } public static void closeReader(Reader reader) { if (null != reader) { try { reader.close(); reader = null; } catch (IOException e) { LOG.error("close reader faile!", e); } } } public static void closeStream(InputStream is, OutputStream os) { if (null != is) { try { is.close(); is = null; } catch (IOException e) { LOG.error("close InputStream fail!", e); } } if (null != os) { try { os.close(); os = null; } catch (IOException e) { LOG.error("close OutputStream fail!", e); } } } }
4. CommonConstants
import java.util.Map; import java.util.concurrent.ConcurrentHashMap; /** * @ClassName: CommonConstants * @Description: 公共常量类 * @author wenjianhai * @company 广州市道一信息技术有限公司 * @date 2012-11-8 * @version V1.0 */ public final class CommonConstants { private CommonConstants() { } /** 存放属性配置文件(*.properties)中的key-value */ public static MappropMap = new ConcurrentHashMap (16); }
5. 初始化配置文件
import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import org.apache.log4j.Logger; /** * @ClassName: InitServlet * @Description: 服务启动时需要初始化 * @author * @company * @date 2012-11-8 * @version V1.0 */ public class InitServlet extends HttpServlet { /** * @Fields serialVersionUID : 序列化id */ private static final long serialVersionUID = -283273527502774557L; private static final Logger log = Logger.getLogger(InitServlet.class); @Override public void init() throws ServletException { log.info("Start to init InitServlet."); // 必须要调用父类的初始化方法 super.init(); // 解析并读取属性文件 ParsePropertiesFile.getInstance().getConfig("database.properties"); log.info("End to init InitServlet."); } }
6. 在 web.xml 中配置servlet
initServlet cn.com.do1.component.init.InitServlet 2 initServlet /initServlet
7. 数据库连接池
import java.sql.Connection; import java.sql.SQLException; import java.util.Vector; import javax.sql.DataSource; import org.apache.commons.dbcp.BasicDataSource; import org.apache.commons.lang.time.StopWatch; import org.apache.log4j.Logger; import com.mchange.v2.c3p0.ComboPooledDataSource; /** * @ClassName: ConnectionPool * @Description: 数据库连接池,用于获取第三方数据库的连接 * @author * @company * @date 2012-11-15 * @version V1.0 */ public final class ConnectionPool { private Vectorpool; private int poolSize;// 连接池大小 private static final Logger log = Logger.getLogger(ConnectionPool.class); // 线程局部变量 protected static ThreadLocal threadLocalCon = new ThreadLocal (); private static final ConnectionPool instance = new ConnectionPool(); private static DataSource dataSource; /* * 私有构造方法,禁止在外部创建本类的对象,要想获得本类的对象,统一通过getInstance()方法。 * 在构造方法中,完成数据池配置文件的加载以及数据池的创建。 */ private ConnectionPool() { try { this.poolSize = Integer.parseInt(CommonConstants.propMap .get("c3p0.maxPoolSize")); if (dataSource == null) { initDataSource(); } if (null == pool) { // 根据配置信息创建连接池 pool = new Vector (poolSize); pool = addConnection(poolSize); } } catch (Exception e) { log.error("Init Connection Pool fail!", e); } } /* * 获取本类的对象 */ public static ConnectionPool getInstance() { return instance; } /* * 释放一条连接到连接池中。 */ public synchronized void release(Connection conn) { // 重新设置成自动提交(使用事务时,会设置为不自动提交,所以释放连接时,重置成事务自动提交) try { conn.setAutoCommit(true); } catch (SQLException e) { log.error("set auto commit fail!", e); } catch (Exception e) { log.error("set auto commit fail!", e); } // 重新放入连接池中 pool.add(conn); // 从线程变量中移除 threadLocalCon.set(null); } /* * 从连接池中获取一个连接 */ public Connection getConnection() { Connection con = threadLocalCon.get(); try { if (con == null || con.isClosed()) { while (pool.size() == 0) { try { Thread.sleep(500); } catch (InterruptedException e) { log.error(e); } } synchronized (this) { con = pool.get(0); threadLocalCon.set(con); } } } catch (Exception e1) { log.error("Get Connection fail!", e1); } return con; } /* * 根据传入的参数,在连接池中新建num个连接。 */ private Vector addConnection(int num) { StopWatch sw = new StopWatch(); sw.start(); Connection conn = null; try { // Class.forName(driverClassName); for (int i = 0; i < num; i++) { // conn = DriverManager.getConnection(url, username, password); conn = dataSource.getConnection(); pool.add(conn); } } catch (Exception e) { log.error("add Connection fail!", e); } sw.stop(); log.info("end add connection,the time-consuming is:" + sw.getTime()); return pool; } /* * 清空连接池中所有连接。 */ public synchronized void closePool() { for (int i = 0; i < pool.size(); i++) { try { pool.get(i).close(); } catch (SQLException e) { log.error("close Connection Pool fail!", e); } catch (Exception e) { log.error("close Connection Pool fail!", e); } pool.remove(i); } } /* * 获取当前连接池中可用连接数 */ public int getAvailableConnectNumber() { return pool.size(); } /** * @Title: initDataSource * @Description:初始化数据源,使用 DPCP * @author * @date 2012-11-16 */ public static void initDataSourceDpcp() { // DPCP 连接池 BasicDataSource ds = new BasicDataSource(); ds.setDriverClassName(CommonConstants.propMap.get("c3p0.driver")); ds.setUrl(CommonConstants.propMap.get("c3p0.url")); ds.setUsername(CommonConstants.propMap.get("c3p0.user")); ds.setPassword(CommonConstants.propMap.get("c3p0.password")); // 设置从数据源中返回的连接是否采用自动提交机制 ds.setDefaultAutoCommit(Boolean.parseBoolean(CommonConstants.propMap .get("c3p0.autoCommitOnClose"))); // 最大连接数据库连接数,设置为0时,表示没有限制 ds.setInitialSize(Integer.parseInt(CommonConstants.propMap .get("c3p0.initialPoolSize"))); // 最大连接数 ds.setMaxActive(Integer.parseInt(CommonConstants.propMap .get("c3p0.maxPoolSize"))); // 在不新建连接的条件下,池中保持空闲的最少连接数 ds.setMinIdle(Integer.parseInt(CommonConstants.propMap .get("c3p0.minPoolSize"))); // 最大等待秒数,单位为毫秒, 超过时间会报出错误信息 ds.setMaxWait(Integer.parseInt(CommonConstants.propMap .get("dpcp.maxwait"))); dataSource = ds; } /** * @Title: initDataSource * @Description:初始化数据源,使用C3P0 * @author * @date 2012-11-16 */ public static void initDataSource() { // C3P0 连接池 ComboPooledDataSource cpds = new ComboPooledDataSource(); try { cpds.setDriverClass(CommonConstants.propMap.get("c3p0.driver")); } catch (Exception e) { log.error("load driver class fail!", e); } cpds.setUser(CommonConstants.propMap.get("c3p0.user")); cpds.setPassword(CommonConstants.propMap.get("c3p0.password")); cpds.setJdbcUrl(CommonConstants.propMap.get("c3p0.url")); // 初始化时获取三个连接,取值应在minPoolSize与maxPoolSize之间。Default: 3 initialPoolSize cpds.setInitialPoolSize(Integer.parseInt(CommonConstants.propMap .get("c3p0.initialPoolSize"))); // 连接池中保留的最大连接数。Default: 15 maxPoolSize cpds.setMaxPoolSize(Integer.parseInt(CommonConstants.propMap .get("c3p0.maxPoolSize"))); // 连接池中保留的最小连接数。 cpds.setMinPoolSize(Integer.parseInt(CommonConstants.propMap .get("c3p0.minPoolSize"))); // 获得连接的最大等待毫秒数。Default: 1000 acquireRetryDelay cpds.setAcquireRetryDelay(Integer.parseInt(CommonConstants.propMap .get("c3p0.acquireRetryDelay"))); // 最大空闲时间,60秒内未使用则连接被丢弃。若为0则永不丢弃。Default: 0 maxIdleTime cpds.setMaxIdleTime(Integer.parseInt(CommonConstants.propMap .get("c3p0.maxIdleTime"))); // 当连接池中的连接耗尽的时候c3p0一次同时获取的连接数。Default: 3 acquireIncrement cpds.setAcquireIncrement(Integer.parseInt(CommonConstants.propMap .get("c3p0.acquireIncrement"))); // 每60秒检查所有连接池中的空闲连接。Default: 0 idleConnectionTestPeriod // cpds.setIdleConnectionTestPeriod(60); // 连接关闭时默认将所有未提交的操作回滚。Default: false autoCommitOnClose cpds.setAutoCommitOnClose(Boolean.parseBoolean(CommonConstants.propMap .get("c3p0.autoCommitOnClose"))); // maxStatementsPerConnection定义了连接池内单个连接所拥有的最大缓存statements数 cpds.setMaxStatementsPerConnection(Integer .parseInt(CommonConstants.propMap .get("c3p0.maxStatementsPerConnection"))); // c3p0是异步操作的,缓慢的JDBC操作通过帮助进程完成,扩展这些操作可以有效的提升性能 .通过多线程实现多个操作同时被执行 cpds.setNumHelperThreads(Integer.parseInt(CommonConstants.propMap .get("c3p0.numHelperThreads"))); // 定义在从数据库获取新连接失败后重复尝试的次数。Default: 30 acquireRetryAttempts cpds.setAcquireRetryAttempts(Integer.parseInt(CommonConstants.propMap .get("c3p0.acquireRetryAttempts"))); // 定义所有连接测试都执行的测试语句。在使用连接测试的情况下这个一显著提高测试速度。注意:测试的表必须在初始数据源的时候就存在。Default: // null preferredTestQuery // cpds.setPreferredTestQuery("select sysdate from dual"); // 因性能消耗大请只在需要的时候使用它。如果设为true那么在每个connection提交的 // 时候都将校验其有效性。建议使用idleConnectionTestPeriod或automaticTestTable // 等方法来提升连接测试的性能。Default: false testConnectionOnCheckout // cpds.setTestConnectionOnCheckout(true); // 如果设为true那么在取得连接的同时将校验连接的有效性。Default: false testConnectionOnCheckin // cpds.setTestConnectionOnCheckin(true); // JDBC的标准参数,用以控制数据源内加载的PreparedStatements数量。但由于预缓存的statements属于单个connection而不是整个连接池。所以设置这个参数需要考虑到多方面的因素。如果maxStatements与maxStatementsPerConnection均为0,则缓存被关闭。Default: // 0 // cpds.setMaxStatements(1); // 获取连接失败将会引起所有等待连接池来获取连接的线程抛出异常。但是数据源仍有效 // 保留,并在下次调用getConnection()的时候继续尝试获取连接。如果设为true,那么在尝试 // 获取连接失败后该数据源将申明已断开并永久关闭。Default: false breakAfterAcquireFailure // cpds.setBreakAfterAcquireFailure(false); dataSource = cpds; } /** * @Title: beginTransaction * @Description: 开始事务 * @param conn * :数据库连接 * @author * @date 2012-11-16 */ public static void beginTransaction(Connection conn) { try { if (conn != null) { if (conn.getAutoCommit()) { // 手动提交 conn.setAutoCommit(false); } } } catch (SQLException e) { log.error("set auto commit to false fail!", e); } catch (Exception e) { log.error("begin Transaction fail!", e); } } /** * @Title: commitTransaction * @Description: 提交事务 * @param conn * :数据库连接 * @author * @date 2012-11-16 */ public static void commitTransaction(Connection conn) { try { if (conn != null) { if (!conn.getAutoCommit()) { conn.commit(); } } } catch (SQLException e) { log.error("commit transaction fail!", e); } catch (Exception e) { log.error("commit transaction fail!", e); } } /** * @Title: rollbackTransaction * @Description: 事务回滚 * @param conn * :数据库连接 * @author * @date 2012-11-16 */ public static void rollbackTransaction(Connection conn) { try { if (conn != null) { if (!conn.getAutoCommit()) { conn.rollback(); } } } catch (SQLException e) { log.error("rollback transaction fail!", e); } catch (Exception e) { log.error("rollback transaction fail!", e); } } /** * @Title: resetConnection * @Description: 事务重置 * @param conn * :数据库连接 * @author * @date 2012-11-16 */ public static void resetConnection(Connection conn) { try { if (conn != null) { if (conn.getAutoCommit()) { conn.setAutoCommit(false); } else { conn.setAutoCommit(true); } } } catch (SQLException e) { log.error("reset transaction fail!", e); } catch (Exception e) { log.error("reset transaction fail!", e); } } /** * @Title: setAutoCommit * @Description: 设置是否自动提交事务 * @param conn * 数据库连接 * @param autoCommit * :是否自动提交 * @author * @date 2012-11-16 */ public static void setAutoCommit(Connection conn, boolean autoCommit) { if (conn != null) { try { conn.setAutoCommit(autoCommit); } catch (SQLException e) { log.error("set Auto Commit fail!", e); } catch (Exception e) { log.error("set Auto Commit fail!", e); } } } }
8. 调用 (DAO层)
/** * @Title: findExpireNotice * @Description: 查找2个月后到期的许可证、工商执照 * @param isAll * 0:查找所有已到期的,1:查找2个月后到期的 * @return List * @throws Exception * @see cn.com.do1.component.business.licencenotice.dao.ILicencenoticeDAO#findExpireNotice(int) * @author */ @Override public ListfindExpireNotice(YcBusinessMessagePO bm, int isAll) throws Exception { // 获取中间库连接 Connection conn = null; try { conn = ConnectionPool.getInstance().getConnection(); } catch (Exception e) { logger.error("连接中间库失败!", e); throw new Exception("连接中间库失败!"); } List list = new ArrayList (); String sql = null; if (0 == isAll) { // 查找所有已到期的 sql = "select b.license_code,a.CUST_NAME,a.CORPORATION_NAME," + "a.MANAGER,a.MANAGER_TEL " + "from rm_cust a,rm_license b where a.cust_id=b.cust_id " + " and status='1' and INVALIDATION_DATE<=current date"; } else if (1 == isAll) { // 查找bm.getWarningLimit()天后到期的 sql = "select b.license_code,a.CUST_NAME,a.CORPORATION_NAME," + "a.MANAGER,a.MANAGER_TEL " + "from rm_cust a,rm_license b where a.cust_id=b.cust_id " + " and status='1' " + " and INVALIDATION_DATE=(current date + " + bm.getWarningLimit() + " days)"; } logger.info("查找2个月后到期的许可证、工商执照,the sql is:\r\n" + sql); PreparedStatement ps = conn.prepareStatement(sql); ResultSet rs = ps.executeQuery(); RmLicense license = null; while (rs.next()) { license = new RmLicense(); license.setLicenseCode(rs.getString("license_code")); license.setCustName(rs.getString("CUST_NAME")); license.setCorporationName(rs.getString("CORPORATION_NAME")); license.setManager(rs.getString("MANAGER")); license.setManagerTel(rs.getString("MANAGER_TEL")); list.add(license); } ps.close(); ConnectionPool.getInstance().release(conn); return list; }