亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java游戲服務器之數據庫表存取封裝

 更新時間:2015年11月30日 15:11:18   作者:Metazion  
這篇文章主要介紹了Java游戲服務器之數據庫表存取封裝的相關資料,需要的朋友可以參考下

項目涉及的數據庫表并不多,但每個select、insert、update和delete都去手動拼接字符串,是很低效的,尤其在時常要修改結構的情況下。開發(fā)的一個目標就是自動化,即能自動實現的事情就不要手動去做;還有一個原則是單一化,即盡量保證數據或邏輯一個入口一個出口。這個需求可以使用一些開源庫解決,但因為需求簡單,目標明確,沒有必要引入多余的第三方庫。于是自己寫了一個,至少滿足當前需求。

數據庫表的封裝,核心類有兩個,表(Table)和記錄(Record)。首先需要一個Table類保存數據庫表結構的描述,并籍此自動生成相應SQL語句。其次需要一個Record類自動設置SQL參數,并從返回結果集中自動生成邏輯對象。

table類表結構描述可以有兩個來源,自動從數據庫獲取,或從配置表加載。這里選擇從配置表加載的方式,一來實現簡單,二來應用面更廣。

下面是一個賬戶表的配置示例(user.xml)。

<Table name="user" primaryKey="user_id" primaryField="userId">
  <Column name="username" field="username" type="2" />
  <Column name="password" field="password" type="2" />
  <Column name="salt" field="salt" type="1" />
  <Column name="reg_time" field="registerTime" type="3" />
  <Column name="last_login_time" field="lastLoginTime" type="3" />
</Table>

只定義了一個主鍵,有需要可對此擴充。每列name對應數據庫表的列名,field對應邏輯對象的成員變量名,type對應字段的類型,比如是int、string、timestamp等,有了名字和類型,就可以使用反射方式自動get和set數據。

Table類讀取配置文件獲得數據表的結構描述。

public class Table<T> {
  public class TableField {
    public static final int TYPE_INTEGER = 1;
    public static final int TYPE_STRING = 2;
    public static final int TYPE_TIMESTAMP = 3;
    public String columnName = "";
    public String fieldName = "";
    public int type = 0;
  }
  private String tableName = "";
  private TableField primaryField = new TableField();
  private ArrayList<TableField> tableFields = new ArrayList<TableField>();
  private String selectAllSql = "";
  private String selectSql = "";
  private String insertSql = "";
  private String updateSql = "";
  private String deleteSql = "";
  ...

然后生成PrepareStatement方式讀寫的select、insert、update和delete的預處理SQL字符串。如update:

private String generateUpdateSql() {
    String sql = "UPDATE " + tableName + " SET ";
    int size = tableFields.size();
    for (int index = 0; index < size; ++index) {
      TableField tableField = tableFields.get(index);
      String conjunction = index == 0 ? "" : ",";
      String colSql = tableField.columnName + " = ?";
      sql = sql + conjunction + colSql;
    }

    sql = sql + " WHERE " + primaryField.columnName + "=?";
    return sql;
  }

Table類的功能就這么多,下面是關鍵的Record類,其使用反射自動存取數據。

public class Record<T> {
  private Table<T> table = null;
  private T object = null;
  ...

模板參數T即一個表記錄對應的邏輯對象。在我們的示例里,即賬戶數據類:

public class UserData implements Serializable {
  // 用戶ID
  public int userId = 0;
  // 用戶名
  public String username = "";
  // 密碼
  public String password = "";
  ...

有了SQL語句,要先設置參數,才能執(zhí)行。主鍵和普通字段分開設置。

 public int setPrimaryParams(int start, PreparedStatement pst) throws Exception {
    Table<T>.TableField primaryField = table.getPrimaryField();
    Object value = getFieldValue(primaryField);
    value = toDBValue(primaryField, value);
    pst.setObject(start, value);
    return start + 1;
  }
  public int setNormalParams(int start, PreparedStatement pst) throws Exception {
    ArrayList<Table<T>.TableField> normalFields = table.getNoramlFields();
    final int size = normalFields.size();
    for (int index = 0; index < size; ++index) {
      Table<T>.TableField tableField = normalFields.get(index);
      Object value = getFieldValue(tableField);
      value = toDBValue(tableField, value);
      pst.setObject(start + index, value);
    }
    return start + size;
  }

就是根據表結構描述,通過反射獲取對應字段的值然后設置。

 private Object getFieldValue(Table<T>.TableField tableField) throws Exception {
    Field field = object.getClass().getDeclaredField(tableField.fieldName);
    return field.get(object);
  }

toDBValue作用是將Java邏輯類型轉成對應數據庫類型,比如時間,在邏輯里是Long,而數據庫類型是Timestamp。

 private Object toDBValue(Table<T>.TableField tableField, Object value) {
    if (tableField.type == TableField.TYPE_TIMESTAMP) {
      value = new Timestamp((long) value);
    }
    return value;
  }

以設置update SQL參數為例:

 public void setUpdateParams(PreparedStatement pst) throws Exception {
    final int start = setNormalParams(1, pst);
    setPrimaryParams(start, pst);
  }

之后執(zhí)行該SQL語句就可以了。如果是select語句還會返回結果集(ResultSet),從結果集自動生成邏輯對象原理類似,算是一個逆過程,詳細參看文末代碼。

下面給出一個使用的完整示例:

private static final Table<UserData> udTable = new Table<UserData>();
...
udTable.load("user.xml");
...
public static boolean updateUserData(UserData userData) {
    boolean result = false;
    Record<UserData> record = udTable.createRecord();
    record.setObject(userData);
    PreparedStatement pst = null;
    try {
      String sql = udTable.getUpdateSql();
      pst = DbUtil.openConnection().prepareStatement(sql);
      record.setUpdateParams(pst);
      result = pst.executeUpdate() > 0;
    } catch (Exception e) {
      e.printStackTrace();
    } finally {
      DbUtil.closeConnection(null, pst);
    }
    return result;
  }

代碼封裝得很簡易,有更多需求可據此改進。

相關文章

  • Java CAS操作與Unsafe類詳解

    Java CAS操作與Unsafe類詳解

    這篇文章主要介紹了Java CAS操作與Unsafe類的相關資料,幫助大家更好的理解和學習使用Java,感興趣的朋友可以了解下
    2021-02-02
  • SpringBoot2零基礎到精通之數據庫專項精講

    SpringBoot2零基礎到精通之數據庫專項精講

    SpringBoot是一種整合Spring技術棧的方式(或者說是框架),同時也是簡化Spring的一種快速開發(fā)的腳手架,本篇我們來學習如何連接數據庫進行操作
    2022-03-03
  • Java核心編程之文件隨機讀寫類RandomAccessFile詳解

    Java核心編程之文件隨機讀寫類RandomAccessFile詳解

    這篇文章主要為大家詳細介紹了Java核心編程之文件隨機讀寫類RandomAccessFile,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • Monaco?Editor實現sql和java代碼提示實現示例

    Monaco?Editor實現sql和java代碼提示實現示例

    這篇文章主要為大家介紹了Monaco?Editor代碼提示sql和java實現示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • Java?Stream常用方法合集(超詳細)

    Java?Stream常用方法合集(超詳細)

    Stream?API?提供了一種更為簡潔高效的的方式來處理集合數據,??可讀性較高,?所以本文為大家整理了Java?Stream中的常用方法,希望對大家有所幫助
    2023-07-07
  • Spring-retry實現循環(huán)重試功能

    Spring-retry實現循環(huán)重試功能

    這篇文章主要介紹了Spring-retry 優(yōu)雅的實現循環(huán)重試功能,通過@Retryable注解,優(yōu)雅的實現循環(huán)重試功能,需要的朋友可以參考下
    2023-07-07
  • SpringBoot整合Retry實現錯誤重試過程逐步介紹

    SpringBoot整合Retry實現錯誤重試過程逐步介紹

    重試的使用場景比較多,比如調用遠程服務時,由于網絡或者服務端響應慢導致調用超時,此時可以多重試幾次。用定時任務也可以實現重試的效果,但比較麻煩,用Spring Retry的話一個注解搞定所有,感興趣的可以了解一下
    2023-02-02
  • Java之ThreadLocal使用常見和方式案例講解

    Java之ThreadLocal使用常見和方式案例講解

    這篇文章主要介紹了Java之ThreadLocal使用常見和方式案例講解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下
    2021-08-08
  • SpringBoot實現PPT格式文件上傳并在線預覽功能

    SpringBoot實現PPT格式文件上傳并在線預覽功能

    本文介紹SpringBoot實現PPT格式文件上傳并在線預覽功能,通過上傳接口,可在C盤的tempfile目錄下找到上傳的文件,預覽時會在同級目錄下創(chuàng)建一個相同文件名后綴為pdf的文件,每次預覽會先查找文件是否存在,存在則直接預覽,不存在則會走上面的處理,需要的朋友可以參考下
    2022-02-02
  • Java?NIO?中?Selector?解析

    Java?NIO?中?Selector?解析

    這篇文章主要介紹了Java?NIO?中?Selector,Selector即選擇器,選擇器提供選擇執(zhí)行已經就緒的任務的能力即為翻譯為多路復用,下面文章對Selector詳細介紹內容,需要的小伙伴可以參考一下
    2022-02-02

最新評論