はじめに
セッション中のみ保持される一時テーブルが別セッションでも同名で作成できるのか試してみました。
結論としては別セッションであれば作成できるようです。
通常テーブルと各セッションの一時テーブルでそれぞれ登録先のスキーマが異なりました。
概要
2個のセッションで同時接続し、トランザクション中に同名テーブル作成を試しました。
各パターンは試した後にテーブルを削除しています。
後セッション 先セッション | 通常テーブル | 一時テーブル |
---|---|---|
通常テーブル | × | ○ |
一時テーブル | ○ | ○ 元々これを確認したかった |
- ×【先】通常テーブル作成→【後】通常テーブル作成
- ○【先】通常テーブル作成→【後】一時テーブル作成
- ○【先】一時テーブル作成→【後】通常テーブル作成
- ○【先】一時テーブル作成→【後】一時テーブル作成
環境
- Windows 10 Home
- PostgreSQL 14.6
1. ×【先】通常テーブル作成→【後】通常テーブル作成
別セッションでテーブル作成する時は待ち状態になります。 commit後はテーブル重複でエラーになります。
セッション1 (通常テーブル作成)
postgres=# begin;
BEGIN
postgres=*# create table testtable (id numeric);
CREATE TABLE
postgres=*#
セッション2 (通常テーブル作成時に待ち状態で待機)
postgres=# begin;
BEGIN
postgres=*# create table testtable (id numeric);
(待ち状態)
セッション1 (commit)
postgres=*# commit;
COMMIT
postgres=#
セッション2 (待ち状態解放、エラー)
postgres=*# create table testtable (id numeric);
ERROR: duplicate key value violates unique constraint "pg_type_typname_nsp_index"
DETAIL: Key (typname, typnamespace)=(testtable, 2200) already exists.
postgres=!#
2. ○【先】通常テーブル作成→【後】一時テーブル作成
作成可能ですがスキーマがpublicとpg_tempに分かれます。
セッション1 (通常テーブルがpublicスキーマに作成)
postgres=# begin;
BEGIN
postgres=*# create table testtable (id numeric);
CREATE TABLE
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
----------+-----------+----------+----------
public | testtable | テーブル | postgres
(1 行)
postgres=*# insert into testtable values(1);
INSERT 0 1
postgres=*#
セッション2 (一時テーブルはpg_temp_4スキーマに作成)
postgres=# begin;
BEGIN
postgres=*# create temporary table testtable (id numeric);
CREATE TABLE
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
-----------+-----------+----------+----------
pg_temp_4 | testtable | テーブル | postgres
(1 行)
postgres=*# insert into testtable values(2);
INSERT 0 1
postgres=*# select * from testtable;
id
----
2
(1 行)
postgres=*# \dt public.*
"public.*"という名前のリレーションは見つかりませんでした。
postgres=*#
セッション1 (commit)
postgres=*# commit;
COMMIT
postgres=#
セッション2 (スキーマの指定無しだと一時テーブルのほうを参照)
postgres=*# \dt public.*
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
----------+-----------+----------+----------
public | testtable | テーブル | postgres
(1 行)
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
-----------+-----------+----------+----------
pg_temp_4 | testtable | テーブル | postgres
(1 行)
postgres=*# select * from public.testtable;
id
----
1
(1 行)
postgres=*# select * from testtable;
id
----
2
(1 行)
postgres=*#
3. ○【先】一時テーブル作成→【後】通常テーブル作成
こちらも作成時にスキーマがpublicとpg_tempに分かれます。
セッション1 (一時テーブル作成)
postgres=# begin;
BEGIN
postgres=*# create temporary table testtable (id numeric);
CREATE TABLE
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
-----------+-----------+----------+----------
pg_temp_3 | testtable | テーブル | postgres
(1 行)
postgres=*# insert into testtable values(1);
INSERT 0 1
postgres=*# select * from testtable;
id
----
1
(1 行)
postgres=*#
セッション2 (通常テーブル作成)
postgres=# begin;
BEGIN
postgres=*# create table testtable (id numeric);
CREATE TABLE
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
----------+-----------+----------+----------
public | testtable | テーブル | postgres
(1 行)
postgres=*# insert into testtable values(2);
INSERT 0 1
postgres=*# select * from testtable;
id
----
2
(1 行)
postgres=*#
セッション2 (先に通常テーブル作成のセッションBをcommit)
postgres=*# commit;
COMMIT
postgres=#
セッション1 (やはりスキーマの指定無しでは一時テーブルが優先されている)
postgres=*# select * from testtable;
id
----
1
(1 行)
postgres=*# select * from public.testtable;
id
----
2
(1 行)
postgres=*#
4. ○【先】一時テーブル作成→【後】一時テーブル作成
一時テーブル作成時にpg_temp_3とpg_temp_4と異なるスキーマに分かれます。
セッション1 (一時テーブル作成)
postgres=# begin;
BEGIN
postgres=*# create temporary table testtable (id numeric);
CREATE TABLE
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
-----------+-----------+----------+----------
pg_temp_3 | testtable | テーブル | postgres
(1 行)
postgres=*# insert into testtable values(1);
INSERT 0 1
postgres=*# select * from testtable;
id
----
1
(1 行)
postgres=*#
セクション2 (一時テーブル作成)
postgres=# begin;
BEGIN
postgres=*# create temporary table testtable (id numeric);
CREATE TABLE
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
-----------+-----------+----------+----------
pg_temp_4 | testtable | テーブル | postgres
(1 行)
postgres=*# insert into testtable values(2);
INSERT 0 1
postgres=*# select * from testtable;
id
----
2
(1 行)
postgres=*#
セクション1 (commit)
postgres=*# commit;
COMMIT
postgres=#
セクション2 (スキーマ指定無しだとセクション2で作成した一時テーブルを参照)
postgres=*# \dt
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
-----------+-----------+----------+----------
pg_temp_4 | testtable | テーブル | postgres
(1 行)
postgres=*# select * from testtable;
id
----
2
(1 行)
postgres=*#
ちなみに別セッションの一時テーブルの存在は確認できますが、selectは出来ないです。
セッション2 (別セッションの一時テーブルの中身は覗けない)
postgres=*# \dt pg_temp_3.*
リレーション一覧
スキーマ | 名前 | タイプ | 所有者
-----------+-----------+----------+----------
pg_temp_3 | testtable | テーブル | postgres
(1 行)
postgres=*# select * from pg_temp_3.testtable;
ERROR: cannot access temporary tables of other sessions
postgres=!#