熱點推薦:
您现在的位置: 電腦知識網 >> 編程 >> MySQL >> 正文

利用mysql事務特性實現並發安全的自增ID示例

2022-06-13   來源: MySQL 
項目中經常會用到自增id比如uid最簡單的方法就是用直接用數據庫提供的AUTO_INCREMENT但是如果用戶量非常大幾千萬幾億然後需要分表存儲的時候呢這種方案就搞不定了所以最好有一個全局的自增ID的生成器不管是否分表都能從生成器中獲取到全局自增的ID

實現方法應該有很多不過所有的方案都需要解決一個問題就是保證在高並發的情景下數據獲取依然正確每次獲取的ID都不會重復

這裡我分享兩種利用mysql的innodb的事務特性來實現的方案一種是實現過了的另一種沒有試驗過不過應該也能走的通

先介紹第一種在數據庫中單獨設置一張表來存儲ID表有兩個字段一個是種類吧一個就是ID

  復制代碼 代碼如下:

  CREATE TABLE auto_id(
idname varchar() NOT NULL DEFAULT
id bigint() NOT NULL DEFAULT COMMENT
primary key(idname)
)ENGINE=Innodb DEFAULT CHARSET=utf;


接下來是一個存儲過程

  復制代碼 代碼如下:

  delimiter //
drop procedure if exists get_increment_id;
create procedure get_increment_id(in idname_in varchar() in small_in bigint out id_out bigint)
begin
declare oldid bigint;
start transaction;
select id into oldid from maibo_auto_id where idname=idname_in for update;
if oldid is NULL then
insert into maibo_auto_id(idnameid) value(idname_in small_in);
set id_out=small_in;
else
update maibo_auto_id set id=id+ where idname=idname_in;
set id_out=oldid+;
end if;
commit;
end;
//


重點是這句select id into oldid from maibo_auto_id where idname=idname_in for update會給相關數據加一個獨占鎖定這時候別的進程如果來讀取該條記錄就會進入等待等待這個進程commit之後再繼續這樣就保證了在並發的情況下不同的進程不會取到相同的值

如果你的前端是用php實現的

只需執行如下兩個sql就可以獲取到這個small參數是定義的是從多少開始自增

  復制代碼 代碼如下:

  $sql = call get_increment_id({$key} {$small} @id);
$ret = $db>getData(select @id);



還有另外一種方法就是利用mysql的auto_increment

先創建一張表表裡邊只有一個自增字段

  復制代碼 代碼如下:

  create table test(
`id` int() NOT NULL AUTO_INCREMENT COMMENT id
primary key (id)
)ENGINE=MyISAM AUTO_INCREMENT= DEFAULT CHARSET=utf;


通過如下兩條sql

  復制代碼 代碼如下:

  UPDATE test SET id = LAST_INSERT_ID(id + );
SELECT LAST_INSERT_ID();


也能解決問題 LAST_INSERT_ID是不用查表的而且只針對當前連接也就是說別的連接的更新不會影響到當前連接的取值

這樣可能每個ID都得弄一張表來維護這也是缺點

具體使用中如何處理就看自己的選擇了
From:http://tw.wingwit.com/Article/program/MySQL/201405/30865.html
    推薦文章
    Copyright © 2005-2022 電腦知識網 Computer Knowledge   All rights reserved.