package com.hongyuan.core;  
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Types;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern; 
import javax.sql.DataSource; 
import com.mysql.jdbc.jdbc2.optional.MysqlDataSource;  
public class DBUtil {     
    private static DataSource dataSource = null;
    static{
        /**
         * 初始化數(shù)據(jù)源,不同的數(shù)據(jù)庫獲取數(shù)據(jù)源的方式不同,可參考相應數(shù)據(jù)庫的說明文檔。
         */
        MysqlDataSource mds=new MysqlDataSource();
        mds.setURL("jdbc:mysql://localhost:3306/test");
        mds.setUser("test");
        mds.setPassword("123456");
        mds.setCharacterEncoding("utf8");
        dataSource=mds;
    }     
    /**
     * 獲取數(shù)據(jù)庫連接
     * @return
     * @throws SQLException
     */
    public static Connection getConnection() throws SQLException {
        return dataSource.getConnection();
    }  
    /**
     * 關(guān)閉數(shù)據(jù)庫連接資源
     * @param conn  
     * @param s
     * @param rs
     * @throws SQLException
     */
    public static void close(Connection conn, Statement s, ResultSet rs){
        try {
            if (rs != null) rs.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if (s != null) s.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        try {
            if (conn != null) conn.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }     
    /**
     * 執(zhí)行數(shù)據(jù)庫查詢語句
     * @param sql       查詢sql,匿名參數(shù)用?表示,命名參數(shù)使用“:參數(shù)名”表示
     * @param params    查詢參數(shù)
     * @return
     * @throws SQLException
     */
    @SuppressWarnings("unchecked")
    public static List<Map<String,Object>> select(Object sql,Object... params) throws SQLException{
        Object result=DBUtil.executeSql(sql,params);
        if(result==null){
            return null;
        }else{
            return (List<Map<String,Object>>)result;
        }
    }     
    /**
     * 執(zhí)行插入
     * @param sql
     * @param params
     * @return
     * @throws SQLException
     */
    public static int insert(Object sql,Object... params) throws SQLException{
        return DBUtil.update(sql, params);
    }     
    /**
     * 執(zhí)行數(shù)據(jù)庫記錄變更語句(增,刪,改)
     * @param sql       查詢sql,匿名參數(shù)用?表示,命名參數(shù)使用“:參數(shù)名”表示
     * @param params    查詢參數(shù)
     * @return
     * @throws SQLException
     */
    public static int update(Object sql,Object... params) throws SQLException{
        Object result=DBUtil.executeSql(sql,params);
        if(result==null){
            return 0;
        }else{
            return (Integer)result;
        }
    }     
    /**
     * 執(zhí)行刪除
     * @param sql
     * @param params
     * @return
     * @throws SQLException
     */
    public static int delete(Object sql,Object... params) throws SQLException{
        return DBUtil.update(sql, params);
    }     
    /**
     * 通用Sql執(zhí)行方法
     * @param sql       查詢sql,匿名參數(shù)用?表示,命名參數(shù)使用“:參數(shù)名”表示
     * @param params    命名參數(shù)
     * @return
     * @throws SQLException
     */
    public static Object executeSql(Object sql, Object... params) throws SQLException { 
        if(sql==null||"".equals(sql.toString().trim())) throw new SQLException("sql語句為空!");         
        //獲取sql語句
        String sqlStr=sql.toString().trim();         
        //處理命名參數(shù)
        if(params!=null&¶ms.length==1&¶ms[0] instanceof Map){
            List<Object> pList=new ArrayList<Object>();
            Map<String,Object> pMap=(Map<String, Object>)params[0];
            Matcher pMatcher = Pattern.compile(":(
//w+)").matcher(sqlStr);
            while(pMatcher.find()){
                String pName=pMatcher.group(1);
                pList.add(pMap.get(pName));
            }             
            sqlStr=pMatcher.replaceAll("?");
            params=pList.toArray();
        }         
        Connection conn = null;
        PreparedStatement ps = null;
        ResultSet rs = null;
        try {
            conn = DBUtil.getConnection();
            ps = conn.prepareStatement(sqlStr);              
            if (null != params) {
                //初始化查詢參數(shù)
                for(int i=0;i<params.length;i++){
                    Object param = params[i];
                    if(param!=null){
                        ps.setObject(i+1,param);
                    }else{
                        ps.setNull(i+1,Types.NULL);
                    }                     
                }
            }             
            //處理結(jié)果集
            boolean isResultSet = ps.execute();
            List<Object> result = new ArrayList<Object>();
            do {
                if (isResultSet) {
                    List<Map<String,Object>> tableData=new ArrayList<Map<String,Object>>();
                    ResultSet resultSet=ps.getResultSet();
                    while(resultSet.next()){
                        Map<String,Object> rowData=new HashMap<String,Object>();
                        for(int i=1;i<=resultSet.getMetaData().getColumnCount();i++){
                            rowData.put(resultSet.getMetaData().getColumnName(i),resultSet.getObject(i));
                        }
                        tableData.add(rowData);
                    }
                    result.add(tableData);
                } else {
                    result.add(new Integer(ps.getUpdateCount()));
                }
            } while ((isResultSet = ps.getMoreResults()) == true || ps.getUpdateCount() != -1);  
            //處理返回結(jié)果
            if (result.size() == 0) {
                return null;
            } else if (result.size() == 1) {
                return result.get(0);
            } else {
                return result;
            }
        } catch (SQLException e) {
            throw new SQLException("無效sql!-->"+sql);
        } finally {
            DBUtil.close(conn, ps, rs);
        }
    }
}