溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

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

發布時間:2020-08-14 08:21:55 來源:ITPUB博客 閱讀:486 作者:孤竹星 欄目:MySQL數據庫

由于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)

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女