asdtiang的博客 感謝blogjava提供的博客交流平臺

          mysql關于自動編號問題 轉載

          Posted on 2010-01-12 17:42 asdtiang 閱讀(3618) 評論(0)  編輯  收藏
          MySql數據庫唯一編號字段(自動編號字段)
          在數據庫應用,我們經常要用到唯一編號,以標識記錄。在MySQL中可通過數據列的 AUTO_INCREMENT屬性來自動生成。MySQL支持多種數據表,每種數據表的自增屬性都有差異,這里將介紹各種數據表里的數據列自增屬性。

          ISAM表

          如果把一個NULL插入到一個AUTO_INCREMENT數據列里去,MySQL將自動生成下一個序列編號。編號從1開始,并1為基數遞增。

          把0插入AUTO_INCREMENT數據列的效果與插入NULL值一樣。但不建議這樣做,還是以插入NULL值為好。

          當插入記錄時,沒有為AUTO_INCREMENT明確指定值,則等同插入NULL值。

          當插入記錄時,如果為AUTO_INCREMENT數據列明確指定了一個數值,則會出現兩種情況,情況一,如果插入的值與已有的編號重復,則會出現出錯信 息,因為AUTO_INCREMENT數據列的值必須是唯一的;情況二,如果插入的值大于已編號的值,則會把該插入到數據列中,并使在下一個編號將從這個 新值開始遞增。也就是說,可以跳過一些編號。

          如果自增序列的最大值被刪除了,則在插入新記錄時,該值被重用。

          如果用UPDATE命令更新自增列,如果列值與已有的值重復,則會出錯。如果大于已有值,則下一個編號從該值開始遞增。

          如果用replace命令基于AUTO_INCREMENT數據列里的值來修改數據表里的現有記錄,即AUTO_INCREMENT數據列出現在了 replace命令的where子句里,相應的AUTO_INCREMENT值將不會發生變化。但如果replace命令是通過其它的PRIMARY KEY OR UNIQUE索引來修改現有記錄的(即AUTO_INCREMENT數據列沒有出現在replace命令的where子句中),相應的 AUTO_INCREMENT值--如果設置其為NULL(如沒有對它賦值)的話--就會發生變化。

          last_insert_id()函數可獲得自增列自動生成的最后一個編號。但該函數只與服務器的本次會話過程中生成的值有關。如果在與服務器的本次會話中尚未生成AUTO_INCREMENT值, 則該函數返回0。

          其它數據表的自動編號機制都以ISAM表中的機制為基礎。

          MyISAM數據表

          刪除最大編號的記錄后,該編號不可重用。

          可在建表時可用“AUTO_INCREMENT=n”選項來指定一個自增的初始值。

          可用alter table table_name AUTO_INCREMENT=n命令來重設自增的起始值。

          可使用復合索引在同一個數據表里創建多個相互獨立的自增序列,具體做法是這樣的:為數據表創建一個由多個數據列組成的PRIMARY KEY OR UNIQUE索引,并把AUTO_INCREMENT數據列包括在這個索引里作為它的最后一個數據列。這樣,這個復合索引里,前面的那些數據列每構成一種 獨一無二的組合,最末尾的AUTO_INCREMENT數據列就會生成一個與該組合相對應的序列編號。
          HEAP數據表

          HEAP數據表從MySQL4.1開始才允許使用自增列。

          自增值可通過CREATE TABLE語句的 AUTO_INCREMENT=n選項來設置。

          可通過ALTER TABLE語句的AUTO_INCREMENT=n選項來修改自增始初值。

          編號不可重用。

          HEAP數據表不支持在一個數據表中使用復合索引來生成多個互不干擾的序列編號。

          BDB數據表

          不可通過CREATE TABLE OR ALTER TABLE的AUTO_INCREMENT=n選項來改變自增初始值。

          可重用編號。

          支持在一個數據表里使用復合索引來生成多個互不干擾的序列編號。

          InnDB數據表

          不可通過CREATE TABLE OR ALTER TABLE的AUTO_INCREMENT=n選項來改變自增初始值。

          不可重用編號。

          不支持在一個數據表里使用復合索引來生成多個互不干擾的序列編號。

          在使用AUTO_INCREMENT時,應注意以下幾點:

          AUTO_INCREMENT是數據列的一種屬性,只適用于整數類型數據列。

          設置AUTO_INCREMENT屬性的數據列應該是一個正數序列,所以應該把該數據列聲明為UNSIGNED,這樣序列的編號個可增加一倍。

          AUTO_INCREMENT數據列必須有唯一索引,以避免序號重復。

          AUTO_INCREMENT數據列必須具備NOT NULL屬性。

          AUTO_INCREMENT數據列序號的最大值受該列的數據類型約束,如TINYINT數據列的最大編號是127,如加上UNSIGNED,則最大為 255。一旦達到上限,AUTO_INCREMENT就會失效。

          當進行全表刪除時,AUTO_INCREMENT會從1重新開始編號。全表刪除的意思是發出以下兩條語句時:

          delete from table_name;
          or
          truncate table table_name

          這是因為進行全表操作時,MySQL實際是做了這樣的優化操作:先把數據表里的所有數據和索引刪除,然后重建數據表。如果想刪除所有的數據行又想保留序列 編號信息,可這樣用一個帶where的delete命令以抑制MySQL的優化:

          delete from table_name where 1;

          這將迫使MySQL為每個刪除的數據行都做一次條件表達式的求值操作。

          強制MySQL不復用已經使用過的序列值的方法是:另外創建一個專門用來生成AUTO_INCREMENT序列的數據表,并做到永遠不去刪除該表的記錄。 當需要在主數據表里插入一條記錄時,先在那個專門生成序號的表中插入一個NULL值以產生一個編號,然后,在往主數據表里插入數據時,利用 LAST_INSERT_ID()函數取得這個編號,并把它賦值給主表的存放序列的數據列。如:

          insert into id set id = NULL;
          insert into main set main_id = LAST_INSERT_ID();

          可用alter命令給一個數據表增加一個具有AUTO_INCREMENT屬性的數據列。MySQL會自動生成所有的編號。

          要重新排列現有的序列編號,最簡單的方法是先刪除該列,再重建該,MySQL會重新生連續的編號序列。

          在不用AUTO_INCREMENT的情況下生成序列,可利用帶參數的LAST_INSERT_ID()函數。如果用一個帶參數的 LAST_INSERT_ID(expr)去插入或修改一個數據列,緊接著又調用不帶參數的LAST_INSERT_ID()函數,則第二次函數調用返回 的就是expr的值。下面演示該方法的具體操作:

          先創建一個只有一個數據行的數據表:
          create table seq_table (id int unsigned not null);
          insert into seq_table values (0);
          接著用以下操作檢索出序列號:
          update seq_table set seq = LAST_INSERT_ID( seq + 1 );
          select LAST_INSERT_ID();
          通過修改seq+1中的常數值,可生成不同步長的序列,如seq+10可生成步長為10的序列。

          該方法可用于計數器,在數據表中插入多行以記錄不同的計數值。再配合LAST_INSERT_ID()函數的返回值生成不同內容的計數值。這種方法的優點 是不用事務或LOCK,UNLOCK表就可生成唯一的序列編號。不會影響其它客戶程序的正常表操作。

          alter table table_name auto_increment=n;
          注意n只能大于已有的auto_increment的整數值,小于的值無效.
          show table status like 'table_name' 可以看到auto_increment這一列是表現有的值.
          步進值沒法改變.只能通過下面提到last_inset_id()函數變通使用


          在使用AUTO_INCREMENT時,應注意以下幾點:

          AUTO_INCREMENT是數據列的一種屬性,只適用于整數類型數據列。

          設置AUTO_INCREMENT屬性的數據列應該是一個正數序列,所以應該把該數據列聲明為UNSIGNED,這樣序列的編號個可增加一倍。

          AUTO_INCREMENT數據列必須有唯一索引,以避免序號重復。

          AUTO_INCREMENT數據列必須具備NOT NULL屬性。

          AUTO_INCREMENT數據列序號的最大值受該列的數據類型約束,如TINYINT數據列的最大編號是127,如加上UNSIGNED,則最大為 255。一旦達到上限,AUTO_INCREMENT就會失效。

          在不用AUTO_INCREMENT的情況下生成序列,可利用帶參數的LAST_INSERT_ID()函數。如果用一個帶參數的 LAST_INSERT_ID(expr)去插入或修改一個數據列,緊接著又調用不帶參數的LAST_INSERT_ID()函數,則第二次函數調用返回 的就是expr的值。下面演示該方法的具體操作:

          先創建一個只有一個數據行的數據表:
          create table seq_table (id int unsigned not null);
          insert into seq_table values (0);
          接著用以下操作檢索出序列號:
          update seq_table set seq = LAST_INSERT_ID( seq + 1 );
          select LAST_INSERT_ID();
          通過修改seq+1中的常數值,可生成不同步長的序列,如seq+10可生成步長為10的序列。

          該方法可用于計數器,在數據表中插入多行以記錄不同的計數值。再配合LAST_INSERT_ID()函數的返回值生成不同內容的計數值。這種方法的優點 是不用事務或LOCK,UNLOCK表就可生成唯一的序列編號。不會影響其它客戶程序的正常表操作。

          有兩點需要加強注意:
          1、只有一列的時候是不行的!
          2、自動編號必須作為主鍵才有效!



          天蒼蒼,野茫茫,風吹草底見牛羊

          只有注冊用戶登錄后才能發表評論。


          網站導航:
           

          posts - 80, comments - 24, trackbacks - 0, articles - 32

          Copyright © asdtiang

          asdtiang的博客 PaidMailz
          點擊廣告網賺A(每天4個廣告,每個0.0025美元,一個搜索廣告0.03美元)
          主站蜘蛛池模板: 漳州市| 阿图什市| 罗定市| 泉州市| 嘉祥县| 武义县| 大宁县| 丹巴县| 新建县| 鹿邑县| 南华县| 绥滨县| 北碚区| 枣庄市| 井研县| 南雄市| 濮阳县| 漯河市| 灯塔市| 竹溪县| 石台县| 屯门区| 广丰县| 兴和县| 宜宾市| 东光县| 彭山县| 巴林右旗| 达尔| 贺州市| 乌拉特中旗| 射洪县| 丰台区| 登封市| 横山县| 财经| 扎赉特旗| 鹤岗市| 庆安县| 梁山县| 普陀区|