大战熟女丰满人妻av-荡女精品导航-岛国aaaa级午夜福利片-岛国av动作片在线观看-岛国av无码免费无禁网站-岛国大片激情做爰视频

專注Java教育14年 全國咨詢/投訴熱線:400-8080-105
動力節(jié)點LOGO圖
始于2009,口口相傳的Java黃埔軍校
首頁 hot資訊 JDBC框架簡單的實現(xiàn)方法

JDBC框架簡單的實現(xiàn)方法

更新時間:2021-06-22 16:23:11 來源:動力節(jié)點 瀏覽1410次

1.準備

  • JDBC基本知識
  • JDBC元數(shù)據(jù)知識
  • 反射基本知識

2.兩個問題

業(yè)務背景:系統(tǒng)中所有實體對象都涉及到基本的CRUD操作。所有實體的CUD操作代碼基本相同,僅僅是發(fā)送給數(shù)據(jù)庫的sql語句不同而已,因此可以把CUD操作的所有相同代碼抽取到工具類的一個update方法中,并定義參數(shù)接收變化的sql語句。

實體的R操作,除sql語句不同之外,根據(jù)操作的實體不同,對ResultSet的映射也各不相同,因此可定義一個query方法,除以參數(shù)形式接收變化的sql語句外,可以使用策略模式由query方法的調(diào)用者決定如何把ResultSet中的數(shù)據(jù)映射到實體對象中。

3.JDBC封裝update query方法

public class JdbcNewUtils {
  private JdbcNewUtils() {}
  /**
   * 這里可以使用properties進行替換
   */
  private static final String USER = "root";
  private static final String PWD = "root";
  private static final String URL = "jdbc:mysql://127.0.0.1:3306/day?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC&generateSimpleParameterMetadata=true";
  private static final String DRIVER= "com.mysql.jdbc.Driver";
  static {
    try {
      Class.forName(DRIVER);
    } catch (ClassNotFoundException e) {
      e.printStackTrace();
    }
  }
  
  public static Connection getConnection() throws SQLException {
    Connection connection = DriverManager.getConnection(URL, USER, PWD);
    return connection;
  }
  /**
   * CUD 返回影響數(shù)目
   * @param sql
   * @param args
   * @return int
   */
  public static int update(String sql,Object [] args) {
    PreparedStatement ps = null;
    Connection conn = null;
    try {
      conn=getConnection();
      ps = conn.prepareStatement(sql);
      for (int i = 1; i <= args.length; i++) {
        ps.setObject(i, args[i-1]);
      }
      return ps.executeUpdate();
    } catch (SQLException e) {
      e.printStackTrace();
    }finally {
      close(conn, ps);
    }
    return 0;
  }
  /**
   * 查詢結果封裝Bean
   * @param sql
   * @param args
   * @param rsh
   * @return Object
   */
  public static Object query(String sql,Object [] args,ResultSetHandler rsh) {
    PreparedStatement ps = null;
    Connection conn = null;
    try {
      conn=getConnection();
      ps = conn.prepareStatement(sql);
      for (int i = 0; i < args.length; i++) {
        ps.setObject(i+1, args[i]);
      }
      return rsh.handle(ps.executeQuery());
    } catch (SQLException e) {
      e.printStackTrace();
    }finally {
      close(conn, ps);
    }
    return null;
  }
  
  /**
  * 關閉所有打開的資源
  */
  public static void close(Connection conn, Statement stmt) {
    if(stmt!=null) {
      try {
        stmt.close();
      } catch (SQLException e) {
        e.printStackTrace();
      }
    }
    if(conn!=null) {
      try {
        conn.close();
      } catch (SQLException e) {
        e.printStackTrace();
      }
    }
  }
  /**
  * 關閉所有打開的資源
  */
  public static void close(Connection conn, Statement stmt, ResultSet rs) {
    if(rs!=null) {
      try {
        rs.close();
      } catch (SQLException e) {
        e.printStackTrace();
      }
    }
    close(conn, stmt);
  }

}

4.query

每次查詢根據(jù)查詢的參數(shù)不同,返回的ResultSet也不同,這個規(guī)則我們需要單獨編寫規(guī)則解析器,這里用到了策略設計模式,

將ResultSetHandler定義解決問題的接口,handle為那些需要實現(xiàn)的具體解決的辦法

public interface ResultSetHandler {
  Object handle(ResultSet resultSet);
}

下面我實現(xiàn)了Beanhandler 和 BeanListHandler 分別是 單個的Bean 和一個列表的Bean

package jdbc.simpleframwork;

import java.lang.reflect.Field;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;

public class BeanHandler implements ResultSetHandler {

  private Class<?> obj;
  public BeanHandler(Class<?> obj) {
    this.obj = obj;
  }
  @Override
  public Object handle(ResultSet resultSet){
    try {
      if(!resultSet.next()) {
        return null;
      }
      Object instance = obj.newInstance();
      ResultSetMetaData metaData = resultSet.getMetaData();
      int count = metaData.getColumnCount();
      for(int i=1;i<=count;i++) {
        Field f = obj.getDeclaredField(metaData.getColumnName(i));
        f.setAccessible(true);
        f.set(instance, resultSet.getObject(i));
      }
      return instance;
    } catch (InstantiationException | IllegalAccessException e) {
      e.printStackTrace();
    } catch (SQLException e) {
      e.printStackTrace();
    } catch (NoSuchFieldException e) {
      e.printStackTrace();
    } catch (SecurityException e) {
      e.printStackTrace();
    }
    return null;
  }
}
package jdbc.simpleframwork;

import java.lang.reflect.Field;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.util.ArrayList;

public class BeanListHandler implements ResultSetHandler {

  private Class<?> clazz;

  public BeanListHandler(Class<?> clazz) {
    super();
    this.clazz = clazz;
  }

  @Override
  public Object handle(ResultSet resultSet) {
    try {
      ArrayList<Object> objlist = new ArrayList<>();
      ResultSetMetaData metaData = resultSet.getMetaData();
      int count = metaData.getColumnCount();
      while (resultSet.next()) {
        Object instace = clazz.newInstance();
        for (int i = 0; i < count; i++) {
          Field f = clazz.getDeclaredField(metaData.getColumnName(i + 1));
          f.setAccessible(true);
          f.set(instace, resultSet.getObject(i + 1));
          f.setAccessible(false);
        }
        objlist.add(instace);

      }
      return objlist;
    } catch (Exception e) {
      e.printStackTrace();
    }
    return null;
  }
}

5.測試

public class TestFramwork {

  public static void main(String[] args) throws SQLException {
    Connection conn = JdbcNewUtils.getConnection();

    String sql = "select * from student where id=?";
    PreparedStatement ps = conn.prepareStatement(sql);
    Student stu = (Student) JdbcNewUtils.query(sql, new Object[] { 1 }, new BeanHandler(Student.class));
    System.out.println(stu);

    String sql2 = "select * from student";
    ArrayList<Student> list = (ArrayList<Student>) JdbcNewUtils.query(sql2, new Object[] {},
        new BeanListHandler(Student.class));
    System.out.println(list);
  }

}

6.總結

Update系列操作:

對于CUD操作,SQL只有站位符號的多少發(fā)生了改變,對于傳遞參數(shù)才是我們需要關注的地方,但是JDBC提供了一系列傳遞參數(shù)解析的辦法,通過set系列函數(shù),將參數(shù)值傳遞進行,所以我們只需要封裝一個通用的update即可

Query系列操作

對R操作,就復雜得多,SQL語句的不同,返回的ResultSet也不同,可以單個Bean或者一個List,一個Map等,可以看出來,實際上很多框架提供的也就是這些方法的封裝

對了真正應用上我們的DAO一邊是下面的寫法

public class AccountDao {
  public void add(Account account) throws SQLException{
    String sql = "insert into account(name , money) values(?, ?)";
    Object[] params = {account.getName(), account.getMoney()};
    JdbcUtils.update(sql, params);
  }
  
  public void delete(int id ) throws SQLException{
    String sql = "delete from account where id = ?";
    Object[] params = {id};
    JdbcUtils.update(sql, params);
  }
  
  public void update(Account account) throws SQLException{
    String sql = "update account set name = ?, money = ? where id = ?";
    Object params[] = {account.getName(), account.getMoney(), account.getId()};
    JdbcUtils.update(sql, params);
  }
  
  public Account find(int id ) throws SQLException{
    String sql = "select * from account where id = ?";
    Object params[] = {id};
    return (Account) JdbcUtils.query(sql, params, new BeanHandler(Account.class));
  }
  
  public List getAll() throws SQLException{
    String sql = "select * from account";
    Object params[] = {};
    return (List)JdbcUtils.query(sql, params, new BeanListHandler(Account.class));
  }
}

以上就是動力節(jié)點小編介紹的"JDBC框架簡單的實現(xiàn)方法",希望對大家有幫助,更詳細的Java基礎教程可以登錄動力節(jié)點官網(wǎng)查看,學習中遇到問題,也可以請教在線咨詢,有專業(yè)老師隨時為您服務。

提交申請后,顧問老師會電話與您溝通安排學習

免費課程推薦 >>
技術文檔推薦 >>
主站蜘蛛池模板: 五月天婷婷免费观看视频在线 | 99久久99 | xx色综合| 欧美香蕉爽爽人人爽观看猫咪 | 天天玩天天操 | 国产伦一区二区三区免费 | 欧美另类日韩中文色综合 | 亚洲第一区在线 | 久久精品视频播放 | 国内毛片视频 | 日本黄色免费一级片 | 涩涩的视频在线观看 | 久国产精品视频 | 精品四虎 | 日本一级片免费观看 | 福利在线网站 | 日韩精品一区二区三区乱码 | 在线观看国产精品日本不卡网 | 色香欲综合成人免费视频 | 久久九九综合 | 欧美区日韩区 | 伊人网站 | 久久视频在线看 | 中文字幕国产 | 天天操穴| 国产精品久久久久9999 | 在线观看精品国语偷拍 | 亚洲一区视频在线 | 亚洲va欧美va国产综合久久 | 精品国产一区二区麻豆 | 久久一区二区精品 | 日本黄色录象 | 日韩最新视频一区二区三 | 福利影院在线 | 中文字幕伦视频 | 91精品国产色综合久久不卡蜜 | 日韩一区二区三区四区 | 全部费免一级毛片不收费 | 亚洲精品国产精品乱码不97 | 免费福利入口在线观看 | 快射视频网 |