jdbc

从28号到31号讲了,从开始连接jdbc,从数据库里面显示数据,然后讲了状态通道,预状态通道,当然存储过程通道老师还没说。。。这个很遗憾。可以考虑自己查一下。

预状态通道比起状态通道具有更高的安全性和更快的执行速度,之所以具有更高的安全性是因为它可以防止sql注入。

这段时间算是一个适应过程,所以 这段时间着重背熟了这个模板,想来也抄了好几遍呢。所以这里面直接给出一个比较靠谱的模板放在这里了。

这个 是前面那个 反射要用到的 一个 基石了,前面调用的就是 后面这个 练级jdbc的模板了。需要给出 jdbc.properties 这个 文件。

jdbc.properties

driver = com.mysql.jdbc.Driver
url = jdbc:mysql://localhost:3306/booksystem
username = root
password = lifei23

 

 

package com.letben.util;
import java.io.IOException;
import java.io.InputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.List;
import java.util.Properties;
/**
 * 希望由调用者执行finally语句
 * @author Administrator
 *(封装应该要把资源都关闭的吧)
 */
public class JDBCTemplate {
    private static String driver ;
    private static String url ;
    private static String user ;
    private static String password ;
    private Connection connection;
    private Statement statement;
    private PreparedStatement preparedStatement;
    private ResultSet resultSet=null;
    static {
        //返回代表这个对象的类或者接口的加载器 
        //getClassLoader()
        //如果是bootstrap,就返回空,对象代表基本数据类型,或者是void的话,也返回空
        //其他状况下,是会返回对应的加载器
        InputStream inputStream = JDBCTemplate.class.
                getClassLoader().getResourceAsStream(
                        "jdbc.properties");
        Properties properties = new Properties();
        try {
            properties.load(inputStream);//加载的时候会报io异常
            driver =properties.getProperty("driver");
            url =properties.getProperty("url");
            user = (String) properties.get("user");
            password = (String) properties.get("password");
            Class.forName(driver);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    /**
     * 对外提供得到链接底层数据库的connection对象
     * @return connection
     * @throws SQLException,由调用者来处理这个异常。因为 我们不希望,由用户来翻我们的底层工具类
     */
    public Connection getConnection() throws SQLException{
        connection = DriverManager.getConnection(url, user, password);
        return connection;
    }
    /**
     * 不想对外提供,本方法作为内部使用,如果我们封装的方法不足以用户调用,希望 调用者通过connection来处理
     * 这样的话,希望在connection里面标记一个参数用以统计被调用次数,借此完善本工具类。
     * 
     * 已经得到 实例化Connection
     */
    private void getStatement(){
        try {
            getConnection();
            statement = connection.createStatement();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    /**
     * 不想对外提供,本方法作为内部使用,如果我们封装的方法不足以用户调用,希望 调用者通过connection来处理
     * 这样的话,希望在connection里面标记一个参数用以统计被调用次数,借此完善本工具类。
     * 已经实例化Connection
     */
    private void getPreparedStatement(String sql){
        try {
            getConnection();
            preparedStatement = connection.prepareStatement(sql);
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    /**
     * 内部方法用以绑定每一个参数和用法,不希望外面调用
     * @param params,需要传一个参数列表进来用以绑定参数
     */
    private void bindParams(List params) {
        try {
            if(params!=null&& !params.isEmpty())
                for(int i=0;i<params.size();i++)
                    preparedStatement.setString(i+1, params.get(i)+"");
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    /**
     * 通过预状态通道,查询数据库,得到结果集。位置参数通过参数列表传入数据库。
     * @param sql 数据库查询语句
     * @param params 参数列表
     * @return ResultSet 结果集
     */
    public ResultSet query(String sql,List params){
        getPreparedStatement(sql);
        try {
            bindParams(params);
            resultSet = preparedStatement.executeQuery();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return resultSet;
    }
    /**
     * 利用 数据库查询语句 得到一个查询结果集,通过状态通道方式
     * @param sql 数据库查询语句
     * @return ResultSet
     */
    public ResultSet query(String sql){
        getStatement();
        try {
            resultSet = statement.executeQuery(sql);
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return resultSet;
    }
    /**
     * 利用数据库查询语句,得到一个更新是否成功,通过状态通道方式
     * @param sql 数据库查询语句
     * @return boolean
     */
    public boolean updateData(String sql){
        getStatement();
        int result=0;
        try {
            result = statement.executeUpdate(sql);
        } catch (SQLException e) {
            e.printStackTrace();
        }
        if(result>0)
            return true;
        return false;
    }
    /**
     * 通过预状态通道 方式更新底层数据库,需要传递数据库语句,和参数列表,对于要查询的语句的参数给出相应的列表,进行操作。
     * @param sql 数据库查询语句
     * @param params 参数列表
     * @return 更新数据 成功与否
     */
    public boolean updateData(String sql,List params){
        getPreparedStatement(sql);
        int result=0;
        bindParams(params);
        try {
            result = preparedStatement.executeUpdate();
        } catch (SQLException e) {
            e.printStackTrace();
        }
        if(result>0)
            return true;
        return false;
    }
    /**
     * 关闭资源操作
     */
    public void closeRes(){
        try {
            if(connection!=null){
                connection.close();
            }
            if(statement!=null){
                statement.close();
            }
            if(preparedStatement!=null){
                preparedStatement.close();
            }
            if(resultSet!=null){
                resultSet.close();
            }
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    //TODO
    //现在有这样一种需求,对于事务的操作的需求,要求一荣俱荣,一损俱损。
    //需要的方法有对应的分别基于两种状态通道的操作,通过形参来重载
    //关闭自动提交,打开自动提交。还有提交一个方法,回滚一个方法。6个方法
    public boolean executeBatch(List<String> sqls){
        boolean flag=false;
        getStatement();
        closeAutoCommit();
        try {
            if(sqls!=null&&!sqls.isEmpty()){
                for (String sql : sqls) {
                    statement.addBatch(sql);
                }
                statement.executeBatch();
            }
            transactionCommit();
            flag=true;
            openAutoCommit();
        } catch (SQLException e) {
            e.printStackTrace();
            transactionRollback();
        }
        return flag;
    }
    /**
     * 【这个的通用性总感觉没有那么强!】
     * 预状态通道
     * @param sql
     * @param params 
     * @return 成组提交结果
     */
    public boolean executeBatch(String sql, String[][] params){
        boolean flag=false;
        getPreparedStatement(sql);
        closeAutoCommit();
        try {
            for (int i = 0; i < params.length; i++) {
                for (int j = 0; j < params[i].length; j++) {
                    preparedStatement.setString(j+1, params[i][j]);
                }
                preparedStatement.addBatch();
            }
            preparedStatement.executeBatch();
            transactionCommit();
            flag=true;
            openAutoCommit();
        } catch (SQLException e) {
            e.printStackTrace();
            transactionRollback();
        }
        return flag;
    }
    /**
     * 事务成组,统一提交
     */
    private void transactionCommit() {
        try {
            connection.commit();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    /**
     * 打开自动执行
     */
    private void openAutoCommit(){
        try {
            connection.setAutoCommit(true);
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    /**
     * 关闭自动执行
     */
    private void closeAutoCommit() {
        try {
            connection.setAutoCommit(false);
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
    /**
     * 事务执行失败回滚
     */
    private void transactionRollback(){
        try {
            connection.rollback();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

 

posted on 2016-02-08 15:28  木鸟飞  阅读(317)  评论(0编辑  收藏  举报

导航