日本国产亚洲-日本国产一区-日本国产一区二区三区-日本韩国欧美一区-日本韩国欧美在线-日本韩国欧美在线观看

當前位置:雨林木風下載站 > 技術開發教程 > 詳細頁面

一個用java完成的數據庫序列生成器

一個用java完成的數據庫序列生成器

更新時間:2022-05-09 文章作者:未知 信息來源:網絡 閱讀次數:

如果是集群等應用這個就不行了,只能在單應用的情況下使用
對于每個序列鍵只需在第一次使用時查詢數據庫,后面的都將不需要查詢
有非常詳細的注釋,我就不多說了


package org.shaoye.common.sql;

import java.util.HashMap;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

/**
* 類<code>Key</code>是一個數據庫主鍵生成器,用序列號的方式來產生數據庫中需要的主鍵值。
* <p>
* <code>Key</code>目前支持的數據庫包括Oracle的所有版本、MySql的3.x以上的版本
* 以及所有支持max()函數的數據庫,支持字段類型僅為數字類型的主鍵,對于字符及其它類型的主鍵尚不提供支持。
* <p>
* 在使用時只需提供表名、字段名(主鍵)以及到數據庫的JDBC連接,如果想要獲得message表的id字段的下一個
* 主鍵值時:
* <p><blockquote><pre>
* java.sql.Connection conn = ...;
* org.shaoye.common.sql.Key key = org.shaoye.common.sql.Key.getInstance();
* int keyValue = key.getNextKey("message", "id", conn);
* String sql = "insert into message (id,...) values (" + keyValue + ",...)";
* //執行插入操作...
* </pre></blockquote><p>
*
* @author 令少爺(shaoye@vip.sina.com)
* @since magic 0.1
*/
public final class Key {

/**
* key的最大值,默認為9223372036854775807,即long類型的最大值
*/
private long max = 9223372036854775807L;

/**
* key的最小值,默認為1
* */
private long min = 1L;

/**
* Key的唯一實例,通過getInstance()方法獲得
* */
private static Key keygen = new Key();

/**
* KeyInfo類的實例列表,默認容量為5個
* */
private HashMap keyList = new HashMap(5); //keyInfo 列表

/**
* 私有的默認構造方法,防止外部構造類的實例
* */
private Key() {
}

/**
* 獲得Key的唯一實例
* */
public static Key getInstance() {
return keygen;
}

/**
* 用指定的表和字段獲得key的下一個值,主鍵的值不得超過2147483647
* @param tableName 數據庫中的表名,表中必須有一個數字主鍵
* @param keyName 表(tableName)中的字段名
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的下一個主鍵的int值
* @throws <code>KeyException</code>如果表名或字段名不存在、訪問數據庫錯誤或key的值大于2147483647時拋出
*/
public int getNextKey(String tableName, String keyName, Connection conn) throws KeyException {
long value = getNextKeyLong(tableName, keyName, conn);
if (value > 2147483647L) {
 throw new KeyException("Key's value too big,please call getNextKeyLong method!");
}
return (new Long(value)).intValue();
}

/**
* 用指定的表和字段獲得key的下一個值,最大為9223372036854775807
* @param tableName 數據庫中的表名,表中必須有一個數字主鍵
* @param keyName 表(tableName)中的字段名
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的下一個主鍵的long值
* @throws <code>KeyException</code>如果表名或字段名不存在或訪問數據庫錯誤時拋出
*/
public long getNextKeyLong(String tableName, String keyName, Connection conn)
throws KeyException {
KeyInfo keyinfo;
String item = tableName + "." + keyName;
try {
 if (keyList.containsKey(item)) {
keyinfo = (KeyInfo) keyList.get(item);
 }
 else {
keyinfo = new KeyInfo(tableName, keyName, conn);
keyList.put(item, keyinfo);
 }
 return keyinfo.getNextKey();
}
catch (SQLException sqle) {
 throw new KeyException(sqle);
}
}

/**
* 用指定的"表<code>.</code>字段"形式的字符串獲得key的下一個值,主鍵的值不得超過2147483647
* @param tableDotField"表.字段"形式的字符串,如:message.id
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的下一個主鍵的int值
* @throws <code>KeyException</code>如果表名或字段名不存在、訪問數據庫錯誤或key的值
* 大于2147483647時拋出
*/
public int getNextKey(String tableDotField, Connection conn) throws KeyException {
long value = getNextKeyLong(tableDotField, conn);
if (value > 2147483647L) {
 throw new KeyException("Key's value too big,please call getNextKeyLong method!");
}
return (new Long(value)).intValue();
}

/**
* 用指定的"表<code>.</code>字段"形式的字符串獲得key的下一個值,最大為9223372036854775807
* @param tableDotField"表.字段"形式的字符串,如:message.id
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的下一個主鍵的int值
* @throws <code>KeyException</code>如果表名或字段名不存在或訪問數據庫錯誤時拋出
*/
public long getNextKeyLong(String tableDotField, Connection conn) throws KeyException {
int dot_index = tableDotField.indexOf(".");
if (tableDotField.indexOf(".") < 1) {
 throw new KeyException("Unknown Key '" + tableDotField + "'!");
}
String tab = tableDotField.substring(0, dot_index);
String key = tableDotField.substring(dot_index);
return getNextKeyLong(tab, key, conn);
}

/**
* 用指定的表和字段獲得key的當前值,主鍵的值不得超過2147483647
* @param tableName 數據庫中的表名,表中必須有一個數字主鍵
* @param keyName 表(tableName)中的字段名
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的當前int值
* @throws <code>KeyException</code>如果表名或字段名不存在、訪問數據庫錯誤或key的值大于2147483647時拋出
*/
public int getCurrentKey(String tableName, String keyName, Connection conn)
throws KeyException {
long value = getCurrentKeyLong(tableName, keyName, conn);
if (value > 2147483647L) {
 throw new KeyException("Key's value too big,please call getCurrentKeyLong method!");
}
return (new Long(value)).intValue();
}

/**
* 用指定的表和字段獲得key的當前值,最大為9223372036854775807
* @param tableName 數據庫中的表名,表中必須有一個數字主鍵
* @param keyName 表(tableName)中的字段名
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的當前long值
* @throws <code>KeyException</code>如果表名或字段名不存在或訪問數據庫錯誤時拋出
*/
public long getCurrentKeyLong(String tableName, String keyName, Connection conn)
throws KeyException {
KeyInfo keyinfo;
String item = tableName + "." + keyName;
try {
 synchronized (keyList) {
if (keyList.containsKey(item)) {
 keyinfo = (KeyInfo) keyList.get(item);
}
else {
 keyinfo = new KeyInfo(tableName, keyName, conn);
 keyList.put(item, keyinfo);
}
 }
 return keyinfo.getCurrentKey();
}
catch (SQLException sqle) {
 throw new KeyException(sqle);
}
}

/**
* 用指定的"表<code>.</code>字段"形式的字符串獲得key的當前值,主鍵的值不得超過2147483647
* @param tableDotField"表.字段"形式的字符串,如:message.id
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的當前int值
* @throws <code>KeyException</code>如果表名或字段名不存在、訪問數據庫錯誤或key的值
* 大于2147483647時拋出
*/
public int getCurrentKey(String tableDotField, Connection conn) throws KeyException {
long value = getCurrentKeyLong(tableDotField, conn);
if (value > 2147483647L) {
 throw new KeyException("Key's value too big,please call getNextKeyLong method!");
}
return (new Long(value)).intValue();
}

/**
* 用指定的"表<code>.</code>字段"形式的字符串獲得key的當前值,最大為9223372036854775807
* @param tableDotField"表.字段"形式的字符串,如:message.id
* @param connJDBC連接,如果欲獲得的key是第一次取值,則必須保證conn能連接到數據庫
* @return key的當前int值
* @throws <code>KeyException</code>如果表名或字段名不存在或訪問數據庫錯誤時拋出
*/
public long getCurrentKeyLong(String tableDotField, Connection conn) throws KeyException {
int dot_index = tableDotField.indexOf(".");
if (tableDotField.indexOf(".") < 1) {
 throw new KeyException("Unknown Key '" + tableDotField + "'!");
}
String tab = tableDotField.substring(0, dot_index);
String key = tableDotField.substring(dot_index);
return getCurrentKeyLong(tab, key, conn);
}
}


/**
* 內部類,用來存儲主鍵信息
* */
class KeyInfo {
private long max = 9223372036854775807L;
private long min = 1L;
private long nextKey;
private String tableName;
private String keyName;
private Connection conn = null;
/**
* keyInfo 對象初始化
*/
KeyInfo(String tableName, String keyName, Connection _conn) throws SQLException {
this.tableName = tableName;
this.keyName = keyName;
this.conn = _conn;
retrieveFromDB();
}
int getMax() {
return (new Long(max)).intValue();
}
long getMaxLong() {
return max;
}
int getMin() {
return (new Long(min)).intValue();
}
long getMinLong() {
return min;
}
/**
* 取下一鍵值
*/
int getNextKey() {
return (new Long(getNextKeyLong())).intValue();
}
/**
* 取下一鍵值
*/
synchronized long getNextKeyLong() {
nextKey++;
return nextKey;
}
/**
* 取當前鍵值
*/
synchronized int getCurrentKey() {
return (new Long(nextKey)).intValue();
}
/**
* 取當前鍵值
*/
synchronized long getCurrentKeyLong() {
return nextKey;
}
/**
* 從數據庫中取當前最大值
*/
void retrieveFromDB() throws SQLException {
PreparedStatement pstmt = null;
ResultSet rs = null;
int keyFromDB = 0;
String sql = "select max(" + keyName + ") from " + tableName;
try {
 pstmt = conn.prepareStatement(sql);
}
catch (Exception ex) {
 throw new KeyException("Can't connect DataBase!");
}
try {
 rs = pstmt.executeQuery();
}
catch (SQLException sqle) {
 if (pstmt != null)
pstmt.close();
 throw new KeyException(
"'" + keyName + "' or '" + tableName + "' isn't exist in DataBase!",
sqle);
}
try {
 if (rs.next()) {
nextKey = rs.getLong(1);
if (nextKey < min) {
 nextKey = min;
}
 }
 else {
nextKey = min;
 }
}
catch (SQLException sqle) {
 throw (sqle);
}
finally {
 if (rs != null)
rs.close();
 if (pstmt != null)
pstmt.close();
}
}
}

溫馨提示:喜歡本站的話,請收藏一下本站!

本類教程下載

系統下載排行

主站蜘蛛池模板: 天天碰视频 | 色久天 | 久久久久青草 | 亚洲欧美日韩精品久久 | 欧美日操 | 日韩精品第一 | 大片免费观看在线视频 | 国产成人综合亚洲怡春院 | 黑人三级 | 国产精品怡红院永久免费 | 一97日本道伊人久久综合影院 | 色婷婷视频在线观看 | 国产一区二区三区福利 | www.怡红院 | 竹菊影视一区二区三区 | 中文字幕人成不卡一区 | 日本高清免费xxx在线观看 | 亚洲另类伦春色综合妖色成人网 | 青青青国产在线观看免费网站 | 国产成人久久久精品一区二区三区 | 99热免费在线观看 | 曰韩一级毛片 | 成人国产精品免费网站 | a级毛片免费网站 | 亚洲欧洲一二三区 | 国产精品成人在线播放 | 性视频高清免费视频播放 | 成人三级精品视频在线观看 | 成人亚洲欧美日韩在线观看 | 精品成人免费一区二区在线播放 | 欧美日韩亚洲国产 | 日本www片| 四虎精品影院 | 91碰在线 | 亚洲精品男人天堂 | 天天操天天操天天射 | 久久机热综合久久国产 | 欧美成人久久久免费播放 | 91尤物国产尤物福利 | 国产精品视频网站你懂得 | 国产欧美日韩精品一区二区三区 |