PostgreSQL中三種自增列sequence,serial,identity區別

来源:https://www.cnblogs.com/wy123/archive/2020/07/24/13367486.html
-Advertisement-
Play Games

這三個對象都可以實現自增,這裡從如下幾個維度來看看這幾個對象有哪些不同,其中功能性上看,大部分特性都是一致的或者類似的。 1,sequence在所有資料庫中的性質都一樣,它是跟具體的欄位不是強綁定的,其特點是支持多個對個對象之間共用。 sequence作為自增欄位值的時候,對錶的寫入需要另外單獨授權 ...


這三個對象都可以實現自增,這裡從如下幾個維度來看看這幾個對象有哪些不同,其中功能性上看,大部分特性都是一致的或者類似的。

1,sequence在所有資料庫中的性質都一樣,它是跟具體的欄位不是強綁定的,其特點是支持多個對個對象之間共用。
 sequence作為自增欄位值的時候,對錶的寫入需要另外單獨授權sequence(GRANT USAGE ON SEQUENCE test_old_id_seq;)
 sequence類型的欄位表,在使用CREATE TABLE new_table LIKE old_table的時候,新表的自增欄位會已久指向原始表的sequence
結論:對於自增欄位,無特殊需求的情況下,sequence不適合作為“自增列”,作為最最次選。

2,identity本質是為了相容標準sql中的語法而新加的,修複了一些serial的缺陷,比如無法通過alter table的方式實現增加或者刪除serial欄位
  2.1 identity定義成generated by default as identity也允許顯式插入,
  2.2 identity定義成always as identity,加上overriding system value也可以顯式不插入
結論:identity是serial的“增強版”,更適合作為“自增列”使用。

3,sequence,serial,identity共同的缺點是在顯式插入之後,無法將自增值更新為表中的最大Id,這一點再顯式插入的情況下是潛在自增欄位Id衝突的
結論:自增列在顯式插入之後,一定要手動重置為表的最大Id。

4,自增欄位的update沒有細看,相對來說自增列的顯式插入是一種常規操作,那些對自增列的update操作,只要腦子沒問題,一般是不會這麼乾的。

參考鏈接:https://www.2ndquadrant.com/en/blog/postgresql-10-identity-columns/

 

原始手稿,懶得整理了,不涉及原理性的東西,動手試一遍就明白了。

---------------------------------------------------------sequence-------------------------------------------------------------
create sequence myschema.seq_1 INCREMENT BY 1 MINVALUE 1 START WITH 1;
create table myschema.test_seq
(
    id int not null default nextval('myschema.seq_1') primary key,
    name varchar(10)
);
隱式插入
insert into myschema.test_seq (name) values ('aaa');
insert into myschema.test_seq (name) values ('bbb');
insert into myschema.test_seq (name) values ('ccc');

select * from myschema.test_seq;

顯式插入
insert into myschema.test_seq (id,name) values (5,'ddd');
select * from test_seq;
再次隱式插入
--可以正常插入
insert into myschema.test_seq (name) values ('eee');
--插入失敗,主鍵重覆,因為序列自身是遞增的,不會關心表中被顯式插入的數據
insert into myschema.test_seq (name) values ('fff');

--重置序列的最大值
select setval('myschema.seq_1',(select max(id) from myschema.test_seq)::BIGINT);
--事務回滾後,序列號並不會回滾
begin;
insert into myschema.test_seq (name) values ('ggg');
rollback;

-- truncate 表之後,序列不受影響
truncate table myschema.test_seq;
--重置序列
ALTER SEQUENCE myschema.seq_1 RESTART WITH 1;
---------------------------------------------------------serial-------------------------------------------------------------
create table myschema.test_serial
(
    id serial primary key,
    name varchar(100)
)

select * from test_serial;

insert into  myschema.test_serial(name) values ('aaa');
insert into  myschema.test_serial(name) values ('bbb');
insert into  myschema.test_serial(name) values ('ccc');

select * from myschema.test_serial;

--顯式插入,可以執行
insert into  myschema.test_serial(id,name) values (5,'ccc');

--再次隱式插入,第二次會報錯,因為隱式插入的話,serial會基於顯式插入之前的Id做自增,serial無法意識到當前已經存在的最大值
insert into  myschema.test_serial(name) values ('xxx');
insert into  myschema.test_serial(name) values ('yyy');
select * from myschema.test_serial;
--truncate table 後serial不會重置
truncate table myschema.test_serial;

insert into  myschema.test_serial(name) values ('aaa');
insert into  myschema.test_serial(name) values ('bbb');
insert into  myschema.test_serial(name) values ('ccc');
select * from myschema.test_serial;
--驗證是否會隨著事務一起回滾,結論:不會
begin;
insert into  myschema.test_serial(name) values ('yyy');
rollback;

--重置serial,需要註意的是重置的Id必須要大於相關表的欄位最大Id,否則會產生重號
SELECT SETVAL((SELECT pg_get_serial_sequence('myschema.test_serial', 'id')), 1, false);


---------------------------------------------------------identity-------------------------------------------------------------
drop table  myschema.test_identiy_1 

create table myschema.test_identiy_1 
(
    id int generated always as identity (cache 100 START WITH 1 INCREMENT BY 1)  primary key , 
    name varchar(100)
);

create table myschema.test_identiy_2
(
    id int generated by default as identity (cache 100 START WITH 1 INCREMENT BY 1)  primary key , 
    name varchar(100)
);


insert into myschema.test_identiy_1(name) values ('aaa');
insert into myschema.test_identiy_1(name) values ('bbb');
insert into myschema.test_identiy_1(name) values ('ccc');


insert into myschema.test_identiy_2(name) values ('aaa');
insert into myschema.test_identiy_2(name) values ('bbb');
insert into myschema.test_identiy_2(name) values ('ccc');


select * from myschema.test_identiy_1;

--顯式插入值,如果定義為generated always as identity則不允許顯式插入,除非增加overriding system value 提示
--一旦提示了overriding system value,可以
insert into myschema.test_identiy_1(id,name) values (5,'ccc');
insert into myschema.test_identiy_1(id,name)overriding system value values (5,'ccc');
select * from myschema.test_identiy_2;
--顯式插入值,如果定義為generated by default as identity則允許顯式插入,
insert into myschema.test_identiy_2(id,name) values (5,'ccc');
--顯式插入後,繼續隱式插入,第二次插入會報錯,identity已久是不識別表中顯式插入後的最大值
insert into myschema.test_identiy_2(name) values ('xxx');
insert into myschema.test_identiy_2(name) values ('yyy');
select * from myschema.test_identiy_2;

總之個identity很扯淡,你定義成always as identity,加上overriding system value可以顯式不插入
定義成generated by default as identity也允許顯式插入
不管怎麼樣,既然都允許顯式插入,那扯什麼淡的來個overriding system value
--truncate後再次插入,自增列不會重置
truncate table myschema.test_identiy_1;
select * from myschema.test_identiy_1;
begin;
insert into myschema.test_identiy_1(name) values ('xxx');
rollback;
--truncate並且RESTART IDENTITY後,會重置自增列
TRUNCATE table myschema.test_identiy_1 RESTART IDENTITY;
select * from myschema.test_identiy_1

--identity自增列的重置表或者更改
ALTER TABLE myschema.test_identiy_1 ALTER COLUMN id RESTART WITH 100;

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一 查看證書 1.1 查看過期時間-方式一 1 [[email protected] ~]# tree /etc/kubernetes/pki/ 2 [[email protected] ~]# for tls in `find /etc/kubernetes/pki -maxdepth 2 -name "*.cr ...
  • SQL 庫結構操作SQL 1、查看所有資料庫 show databases; 2、切換使用資料庫 use 資料庫名; 3、創建資料庫 create database 資料庫名; create database 資料庫名 charset 'utf8'; 4、刪除資料庫 drop database 數據 ...
  • 軟體下載地址:https://www.mongodb.com/try/download/community 提供了二進位執行版,不需要make安裝,將解壓好的內容,直接移動到軟體安裝目錄即可。 在軟體安裝目錄(/usr/local/src)下創建一個名為mongodb的軟連接 ln -s mongo ...
  • 資料庫 多個集合可以組成資料庫。一個MongoDB實例可以承載多個資料庫,他們之間完全獨立。 MongoDB中的資料庫和MySQL中的資料庫概念類似,只是無需創建。 一個資料庫中可以有多個集合,一個集合中可以有多個文檔。 集合 集合就是一組文檔,多個文檔組成一個集合,集合類似於MySQL裡面的表。 ...
  • 基本簡介 MongoDB是一個介於關係資料庫和非關係資料庫之間的產品,是非關係資料庫當中功能最豐富,最像關係資料庫的,語法有點類型javascript面向對象的查詢語言,它是一個買你想幾核的,模式自由的文檔型資料庫。實現類似關係資料庫單表查詢的絕大部門功能,而且還支持對資料庫建立素偶姻。它的特點是高 ...
  • -- 危險操作,處理前記得先備份資料庫 1 declare @sql varchar(500),@tbname varchar(100) 2 begin 3 4 -- 創建游標 5 declare cursor_item cursor fast_forward for select [name] f ...
  • 當我們剛開始接觸一些已經成型的項目時,不複雜還好,複雜的話,比如說ERP項目,其中業務邏輯複雜可能會各種存儲過程之間來回調用,我們可以用 --查詢哪裡調用該表或存儲過程 select distinct object_name(id) from syscomments where id in (sel ...
  • 想把mlsql卸載了重裝,看了許多文章試了很多方法都沒辦法完全卸載,直到看到了這篇文章, 可以完全卸載mysql,在這裡謝謝博主,也拿出來分享給大家 原文鏈接:https://blog.csdn.net/qq_41140741/article/details/81489531 快捷鍵win+r輸入r ...
一周排行
    -Advertisement-
    Play Games
  • 比如要拆分“呵呵呵90909086676喝喝999”,下麵當type=0返回的是中文字元串“呵呵呵,喝喝”,type=1返回的是數字字元串“90909086676,999”, private string GetStrings(string str,int type=0) { IList<strin ...
  • Swagger一個優秀的Api介面文檔生成工具。Swagger可以可以動態生成Api介面文檔,有效的降低前後端人員關於Api介面的溝通成本,促進項目高效開發。 1、使用NuGet安裝最新的包:Swashbuckle.AspNetCore。 2、編輯項目文件(NetCoreTemplate.Web.c ...
  • 2020 年 7 月 30 日, 由.NET基金會和微軟 將舉辦一個線上和為期一天的活動,包括 微軟 .NET 團隊的演講者以及社區的演講者。本次線上大會 專註.NET框架構建微服務,演講者分享構建和部署雲原生應用程式的最佳實踐、模式、提示和技巧。有關更多信息和隨時瞭解情況:https://focu... ...
  • #abp框架Excel導出——基於vue #1.技術棧 ##1.1 前端採用vue,官方提供 UI套件用的是iview ##1.2 後臺是abp——aspnetboilerplate 即abp v1,https://github.com/aspnetboilerplate/aspnetboilerp ...
  • 前言 本文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 作者:碧茂大數據 PS:如有需要Python學習資料的小伙伴可以加下方的群去找免費管理員領取 input()輸入 Python提供了 input() 內置函數從標準輸入讀入一 ...
  • 從12年到20年,python以肉眼可見的趨勢超過了java,成為了當今It界人人皆知的編程語言。 python為什麼這麼火? 網路編程語言搜索指數 適合初學者 Python具有語法簡單、語句清晰的特點,這就讓初學者在學習階段可以把精力集中在編程對象和思維方法上。 大佬都在用 Google,YouT ...
  • 在社會上存在一種普遍的對培訓機構的學生一種歧視的現象,具體表現在,比如:當你去公司面試的時候,一旦你說了你是培訓機構出來的,那麼基本上你就涼了,那麼你瞞著不說,然後又通過了面試成功入職,但是以後一旦在公司被髮現有培訓經歷,可能會面臨被降薪,甚至被辭退,培訓機構出來的學生,在用人單位眼裡就是能力低下的 ...
  • from typing import List# 這道題看了大佬寫的代碼,經過自己的理解寫出來了。# 從最外圍的四周找有沒有為O的,如果有的話就進入深搜函數,然後深搜遍歷# 判斷上下左右的位置是否為Oclass Solution: def solve(self, board: List[List[s ...
  • import requests; import re; import os; # 1.請求網頁 header = { "user-agent":'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_5) AppleWebKit/537.36 (KHTML, li ...
  • import requests; import re; import os; import parsel; 1.請求網頁 header = { "user-agent":'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_5) AppleWebKit/537. ...