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
  • 一、引言:什麼是 JSON JSON (Java Script Object Notation) 是一種很常用的數據格式,它常常用在 web 應用程式中。它可以表示結構化的數據。 下麵是常見的 JSON 文件結構 { "name": "Kamishiro Rize", "age": "22", "o ...
  • 前言 大家好,我是蝸牛,在上一篇中,我們介紹了不同版本的HTTP區別和發展背景,這篇文章我們來聊聊HTTP的缺點,HTTP缺點大致總結有以下三點: 通信使用明文(不加密),內容可能會被竊聽。 不驗證通信方的身份,因此有可能遭遇偽裝(客戶端和服務端都有可能) 無法證明報文的完整性,有可能會被篡改。 其 ...
  • resultMap處理欄位和屬性的映射關係 如果欄位名與實體類中的屬性名不一致,該如何處理映射關係? 第一種方法:為查詢的欄位設置別名,和屬性名保持一致 下麵是實體類中的屬性名: private Integer empId; private String empName; private Integ ...
  • 大家在看到這篇文章前,為了有一個舒適的c++IDE,一定感受到了Dev-c++的廉價感,Clion功能的多餘,VS的臃腫。他們也有自己的優點,但糟點太多,令人十分難受。而VS Code,可以取長補短。下麵的配置內容,可以讓你在刷題時,享受絲滑的動畫,體會集成終端的方便,讓你覺得Coding不再枯燥。 ...
  • 給定一個不含重覆數字的數組 nums ,返回其 所有可能的全排列 。你可以 按任意順序 返回答案。 示例 1: 輸入:nums = [1,2,3] 輸出:[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]] 示例 2: 輸入:nums = [0,1] 輸 ...
  • 設計模式的目的 編寫軟體過程中,程式員面臨著來自 耦合性,內聚性以及可維護性,可擴展性,重用性,靈活性 等多方面的 挑戰,設計模式是為了讓程式(軟體),具有更好 代碼重用性 (即:相同功能的代碼,不用多次編寫) 可讀性 (即:編程規範性, 便於其他程式員的閱讀和理解) 可擴展性 (即:當需要增加新的 ...
  • 本文講解了決策樹的創鍵的過程,包括熵,信息增益的計算,還有決策樹的創建,以及使用matplotlib讓決策樹可視化的詳細過程 ...
  • ♠ use C++11 倍數 若 $a,b,k \in \mathbb N$,且 $a \times k=b$,那麼 $b$ 是 $a$ 的倍數,稱 $a$ 整除 $b$,記作 $a \mid b$。 $[1,n]\in \mathbb N$ 中 $x \in \mathbb N$ 的倍數有 $\l ...
  • LinkList可以定義指向List的指針 1.當函數參數為LinkList L時,意味著只改變或操作List的內容,而不需要改變L這個指針 如 Status GetElem(LinkList L,int i,ElemType) 2.當參數為LinkList &L時,意味著需要改變或操作L這個指針本 ...
  • Spring 5框架 一、Spring概念 1、Spring是輕量級的JavaEE框架 2、Spring可以解決企業應用開發的複雜性 3、Spring有兩個核心部分:IOC和AOP ​ 1)IOC:控制反轉,把創建對象過程交給Spring進行管理 ​ 2)AOP:面向切麵,不修改源代碼進行功能增強 ...