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 [root@master01 ~]# tree /etc/kubernetes/pki/ 2 [root@master01 ~]# 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
  • Timer是什麼 Timer 是一種用於創建定期粒度行為的機制。 與標準的 .NET System.Threading.Timer 類相似,Orleans 的 Timer 允許在一段時間後執行特定的操作,或者在特定的時間間隔內重覆執行操作。 它在分散式系統中具有重要作用,特別是在處理需要周期性執行的 ...
  • 前言 相信很多做WPF開發的小伙伴都遇到過表格類的需求,雖然現有的Grid控制項也能實現,但是使用起來的體驗感並不好,比如要實現一個Excel中的表格效果,估計你能想到的第一個方法就是套Border控制項,用這種方法你需要控制每個Border的邊框,並且在一堆Bordr中找到Grid.Row,Grid. ...
  • .NET C#程式啟動閃退,目錄導致的問題 這是第2次踩這個坑了,很小的編程細節,容易忽略,所以寫個博客,分享給大家。 1.第一次坑:是windows 系統把程式運行成服務,找不到配置文件,原因是以服務運行它的工作目錄是在C:\Windows\System32 2.本次坑:WPF桌面程式通過註冊表設 ...
  • 在分散式系統中,數據的持久化是至關重要的一環。 Orleans 7 引入了強大的持久化功能,使得在分散式環境下管理數據變得更加輕鬆和可靠。 本文將介紹什麼是 Orleans 7 的持久化,如何設置它以及相應的代碼示例。 什麼是 Orleans 7 的持久化? Orleans 7 的持久化是指將 Or ...
  • 前言 .NET Feature Management 是一個用於管理應用程式功能的庫,它可以幫助開發人員在應用程式中輕鬆地添加、移除和管理功能。使用 Feature Management,開發人員可以根據不同用戶、環境或其他條件來動態地控制應用程式中的功能。這使得開發人員可以更靈活地管理應用程式的功 ...
  • 在 WPF 應用程式中,拖放操作是實現用戶交互的重要組成部分。通過拖放操作,用戶可以輕鬆地將數據從一個位置移動到另一個位置,或者將控制項從一個容器移動到另一個容器。然而,WPF 中預設的拖放操作可能並不是那麼好用。為瞭解決這個問題,我們可以自定義一個 Panel 來實現更簡單的拖拽操作。 自定義 Pa ...
  • 在實際使用中,由於涉及到不同編程語言之間互相調用,導致C++ 中的OpenCV與C#中的OpenCvSharp 圖像數據在不同編程語言之間難以有效傳遞。在本文中我們將結合OpenCvSharp源碼實現原理,探究兩種數據之間的通信方式。 ...
  • 一、前言 這是一篇搭建許可權管理系統的系列文章。 隨著網路的發展,信息安全對應任何企業來說都越發的重要,而本系列文章將和大家一起一步一步搭建一個全新的許可權管理系統。 說明:由於搭建一個全新的項目過於繁瑣,所有作者將挑選核心代碼和核心思路進行分享。 二、技術選擇 三、開始設計 1、自主搭建vue前端和. ...
  • Csharper中的表達式樹 這節課來瞭解一下表示式樹是什麼? 在C#中,表達式樹是一種數據結構,它可以表示一些代碼塊,如Lambda表達式或查詢表達式。表達式樹使你能夠查看和操作數據,就像你可以查看和操作代碼一樣。它們通常用於創建動態查詢和解析表達式。 一、認識表達式樹 為什麼要這樣說?它和委托有 ...
  • 在使用Django等框架來操作MySQL時,實際上底層還是通過Python來操作的,首先需要安裝一個驅動程式,在Python3中,驅動程式有多種選擇,比如有pymysql以及mysqlclient等。使用pip命令安裝mysqlclient失敗應如何解決? 安裝的python版本說明 機器同時安裝了 ...