在MySQL中建立實現自增的序列(Sequence)

孤竹星發表於2016-03-21

由於mysql和oracle不太一樣,不支援直接的sequence,所以需要建立一張table來模擬sequence的功能,理由sql語句如下:
第一步:建立--Sequence 管理表

DROP TABLE IF EXISTS sequence;
CREATE TABLE sequence (
     name VARCHAR(50) NOT NULL,
     current_value INT NOT NULL,
     increment INT NOT NULL DEFAULT 1,
     PRIMARY KEY (name)
) ENGINE=InnoDB;


 
第二步:建立--取當前值的函式

DROP FUNCTION IF EXISTS currval;
DELIMITER $
CREATE FUNCTION currval (seq_name VARCHAR(50))
     RETURNS INTEGER
     LANGUAGE SQL
     DETERMINISTIC
     CONTAINS SQL
     SQL SECURITY DEFINER
     COMMENT ''
BEGIN
     DECLARE value INTEGER;
     SET value = 0;
     SELECT current_value INTO value
          FROM sequence
          WHERE name = seq_name;
     RETURN value;
END
$
DELIMITER ;


 
第三步:建立--取下一個值的函式

DROP FUNCTION IF EXISTS nextval;
DELIMITER $
CREATE FUNCTION nextval (seq_name VARCHAR(50))
     RETURNS INTEGER
     LANGUAGE SQL
     DETERMINISTIC
     CONTAINS SQL
     SQL SECURITY DEFINER
     COMMENT ''
BEGIN
     UPDATE sequence
          SET current_value = current_value + increment
          WHERE name = seq_name;
     RETURN currval(seq_name);
END
$
DELIMITER ;


第四步:建立--更新當前值的函式

DROP FUNCTION IF EXISTS setval;
DELIMITER $
CREATE FUNCTION setval (seq_name VARCHAR(50), value INTEGER)
     RETURNS INTEGER
     LANGUAGE SQL
     DETERMINISTIC
     CONTAINS SQL
     SQL SECURITY DEFINER
     COMMENT ''
BEGIN
     UPDATE sequence
          SET current_value = value
          WHERE name = seq_name;
     RETURN currval(seq_name);
END
$
DELIMITER ;


 
第五步:測試函式功能
當上述四步完成後,可以用以下資料設定需要建立的sequence名稱以及設定初始值和獲取當前值和下一個值。

  • INSERT INTO sequence VALUES ('TestSeq', 0, 1);----新增一個sequence名稱和初始值,以及自增幅度
  • SELECT SETVAL('TestSeq', 10);---設定指定sequence的初始值
  • SELECT CURRVAL('TestSeq');--查詢指定sequence的當前值
  • SELECT NEXTVAL('TestSeq');--查詢指定sequence的下一個值


 
在java程式碼中,可直接建立sql語句查詢下一個值,這樣就解決了流水號唯一的問題。
貼出部分程式碼(已測試透過)

public void testGetSequence() {
  Connection conn = JDBCUtils.getConnection(url, userName, password);
  String sql = "SELECT CURRVAL('TestSeq');";
  PreparedStatement ptmt = null;
  ResultSet rs = null;
  try {
    ptmt = conn.prepareStatement(sql);
    rs = ptmt.executeQuery();
    int count = 0;
    while (rs.next()) {
      count = rs.getInt(1);
    }
    System.out.println(count);
  } catch (SQLException e) {
    e.printStackTrace();
  } finally {
    JDBCUtils.close(rs, ptmt, conn);
  }
}


ps:在應用中,還有一種用java程式碼去實現模擬自增sequence的方式,具體思路是建立一張存放sequence的table,然後透過java呼叫sql語句去查詢和修改這個table中指定sequence名稱的值,這種方式請加上synchronized。具體程式碼這裡就不上傳了,因為實現了,未去測試過。


在 oracle 中, sequence 提供多表多欄位可共用一個不重複值。 Mysql 中存在自增列,基本可以滿足 PK 的要求。但自增列存在限制:

a. 只能用於表中的一個欄位,一張不能同時存在兩個以上的自增列 ;

b. 自增列必須被定義為 key ( PK 或 FK ) ;

c. 自增列不能被多個表共用 ;

d. 當 insert 語句不包括自增欄位或將其值設定為 NULL 時,該值會自動填上。

在不要求欄位順序遞增的情況下,可以在 Mysql 中實現序列,再來看下面一個例子:

DROP TABLE IF EXISTS sequence;
  
-- 建sequence表,指定seq列為無符號大整型,可支援無符號值:0(default)到18446744073709551615(0到2^64–1)。
CREATE TABLE sequence (
   name       VARCHAR(50) NOT NULL,
     current_value   BIGINT UNSIGNED NOT NULL DEFAULT 0,
     increment     INT NOT NULL DEFAULT 1,
     PRIMARY KEY (name)  -- 不允許重複seq的存在。
) ENGINE=InnoDB;
  
  
DELIMITER /
  
DROP FUNCTION IF EXISTS currval /
  
CREATE FUNCTION currval(seq_name VARCHAR(50))
RETURNS BIGINT
BEGIN
     DECLARE value BIGINT;
     SELECT current_value INTO value
     FROM sequence
     WHERE upper(name) = upper(seq_name); -- 大小寫不區分.
     RETURN value;
END;
/
  
DELIMITER ;
  
  
DELIMITER /
  
DROP FUNCTION IF EXISTS nextval /
  
CREATE FUNCTION nextval (seq_name VARCHAR(50))
RETURNS BIGINT
BEGIN
     DECLARE value BIGINT;
     UPDATE sequence
     SET current_value = current_value + increment
     WHERE upper(name) = upper(seq_name);
     RETURN currval(seq_name);
END;
/
  
DELIMITER ;
  
DELIMITER /
  
DROP FUNCTION IF EXISTS setval /
  
CREATE FUNCTION setval (seq_name VARCHAR(50), value BIGINT)
RETURNS BIGINT
BEGIN
     UPDATE sequence
     SET current_value = value
     WHERE upper(name) = upper(seq_name);
     RETURN currval(seq_name);
END;
/
  
DELIMITER ;


 在 SQL 中使用序列:
建立序列,往sequence表插入值即可:
mysql> insert into sequence set name='myseq';


檢視當前已建序列:
mysql> select * from sequence;

+-------+---------------+-----------+
| name | current_value | increment |
+-------+---------------+-----------+
| myseq |       0 |     1 |
+-------+---------------+-----------+
1 row in set (0.00 sec)


獲得序列的下一個值,第一次使用,因此值為1:

mysql> select nextval('myseq');

+------------------+
| nextval('myseq') |
+------------------+
|        1 |
+------------------+
1 row in set (0.00 sec)

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/28282660/viewspace-2061242/,如需轉載,請註明出處,否則將追究法律責任。

相關文章