從模板模式到JdbcTemplate

上一篇《模板方法模式java

原始的jdbc

 

關於原始的jdbc,以下:
 mysql

import java.sql.Connection;  
import java.sql.DriverManager;  
import java.sql.ResultSet;  
import java.sql.SQLException;  
import java.sql.Statement;  
  
public class JDBC_Test2 {  
  
    public static void main(String[] args) {  
          
        Connection conn=null;  
        Statement st=null;  
        ResultSet rs=null;  
        String url="jdbc:mysql://localhost:3306/webexample?useUnicode=true&characterEncoding=UTF-8";  
        String userName="root";  
        String passWord="";  
        try {  
            Class.forName("com.mysql.jdbc.Driver");  
            conn=DriverManager.getConnection(url,userName,passWord);  
            st=conn.createStatement();  
            rs=st.executeQuery("select * from admininfo");  
            while (rs.next()) {  
                System.out.println(rs.getString("Aname"));  
            }  
              
              
        } catch (ClassNotFoundException e) {  
            e.printStackTrace();           //log4j 具體記錄  
        }catch (SQLException e) {  
            e.printStackTrace();  
        }  
        finally{  
            try {  
                if (rs!=null) {  
                    rs.close();  
                    rs=null;  
                }  
                if (st!=null) {  
                    st.close();  
                    st=null;  
                }  
                if (conn!=null) {  
                    conn.close();  
                    conn=null;  
                }  
                  
            } catch (SQLException e) {  
                // TODO Auto-generated catch block  
                e.printStackTrace();  
            }  
        }  
    }  
}

我假定朋友們都有必定的編碼經驗,那麼你們再看了最基本的jdbc後,內心應該會有兩個想法
1 做爲jdbc hello world級別的示例程序,上面的代碼寫的仍是很不錯的,該try catch的地方都注意到了
2 若是要大規模使用的話,就比較麻煩了,操做數據庫的核心的代碼不多,可是核心以前以後的操做卻不少。
jdbc鏈接數據庫 獲取數據能夠分爲一下幾步:
1 加載驅動 Class.forName("com.mysql.jdbc.driver");
2 得到鏈接  Connection con=DriverManager.getConnection(url,userName,password);
3 得到Statement  Statement st=con.createStatement();
4 執行sql並得到ResultSet rs=st.executeQuery(sql);
5 處理rs中的數據
             while (rs.next()) {  
                System.out.println(rs.getString("Aname"));  
            }  
6 處理異常
7 關閉鏈接web

仔細看看,其實只有第五步是真正核心的,以前以後的都是那種萬年不變的代碼sql


使用模板模式

 

咱們用模板模式來改裝一下。
 數據庫

package templatemethod1;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;

public abstract class JDBCTemplate  {

	public Object execute(String sql) {
		String  url="";
		String  userName="";
		String  password="";
		Connection con=null;
		Statement st=null;
		ResultSet rs=null;
		try{
			Class.forName("com.mysql.jdbc.driver");
			con=DriverManager.getConnection(url,userName,password);
			st=con.createStatement();
			rs=st.executeQuery(sql);
			//只把核心的第五步 抽象出來 交給子類處理
			Object object=doResultSet(rs); 
			return object;                 
			
		}
		catch (ClassNotFoundException e){
			e.printStackTrace();
		}
		catch(SQLException e){
			e.printStackTrace();
		}
		finally{
			try{
			if(rs!=null){
				rs.close();
				rs=null;
			}
			//...
			}catch(SQLException e){
				e.printStackTrace();

			}
		}
		return null;
	}
	//抽象出來 交給子類處理
	public abstract Object doResultSet(ResultSet rs);
	
}
package templatemethod1;

import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;

import model.Student;


public class JDBCTemplateStudentImpl extends JDBCTemplate {

    //doResultSet 就是關於獲取Student的真正核心的代碼
	@Override
	public Object doResultSet(ResultSet rs) {
		List<Student> userList = new ArrayList<Student>();  
        
        try {  
        	Student student = null;  
            while (rs.next()) {  
  
            	student = new Student();  
            	student.setId(rs.getInt("id"));  
            	student.setBirth(rs.getDate("birth"));  
                userList.add(student);  
            }  
            return userList;  
        } catch (SQLException e) {  
            e.printStackTrace();  
            return null;  
        }  
    }  
	
}

測試代碼ide

package templatemethod1;

import java.util.List;
import model.Student;

public class TemplateMethodTest {
	
	public static void main(String[] args) {
		String  sql="select * from user";
		JDBCTemplate template=new JDBCTemplateStudentImpl();
		List<Student> students=(List<Student>) template.execute(sql);
	}

}

model.Student的代碼我就不給出了,就是幾個字段,getset方法而已。測試

回調

上面的模板方法OK不?
好着呢。
問題是,若是JDBCTemplate中有不少個抽象方法,那麼咱們就得從新不少方法。太累。編碼

那咱們可否只覆蓋咱們想要的呢?
這裏就牽扯到一個名詞,callback(回調)
所謂回調,就是方法參數中傳遞一個接口,父類在調用此方法時,必須調用方法中傳遞的接口的實現類。
咱們直接看代碼url

package templatemethod2;


import java.sql.SQLException;
import java.sql.Statement;


public interface StatementCallback {  
    //這個doInStatement至關於上面的doResultSet
    Object doInStatement(Statement stmt) throws SQLException;  
}
package templatemethod2;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;


public  class JDBCTemplate  {

	public Object execute(StatementCallback action) {
		String  url="";
		String  userName="";
		String  password="";
		Connection con=null;
		Statement st=null;
		ResultSet rs=null;
		try{
			Class.forName("com.mysql.jdbc.driver");
			con=DriverManager.getConnection(url,userName,password);
			st=con.createStatement();
			Object object=action.doInStatement(st);
			return object;			
		}
		//省略try catch
	
		return null;
	}


	public  Object query(StatementCallback action){
		return execute(action);
	}
	
}
下面的測試方法,只要給定一個sql,就能傳回List<Student>
	@SuppressWarnings("unchecked")
	public List<Student> test2(final String sql) {


		JDBCTemplate jdbcTemplate = new JDBCTemplate();
		return (List<Student>) jdbcTemplate.execute(new StatementCallback() {
			@Override
			public Object doInStatement(Statement stmt) throws SQLException {
				ResultSet rs = stmt.executeQuery(sql);
				List<Student> userList = new ArrayList<Student>();

				Student user = null;
				while (rs.next()) {
					user = new Student();
					user.setId(rs.getInt("id"));
					user.setBirth(rs.getDate("birth"));
					userList.add(user);
				}
				return userList;
			}
		});

	}
相關文章
相關標籤/搜索