列がカタログに基づく整数ではなくシリアルデータ型として定義されているかどうかを確認するにはどうすればよいですか?


9

したがって、私は現在、テーブル定義を構築するためにpostgres(9.1)カタログを読むためのSQLをいくつか作成しています。ただし、SERIAL / BIGSERIALデータ型で問題が発生しています。

例:

CREATE TABLE cruft.temp ( id BIGSERIAL PRIMARY KEY );
SELECT * FROM information_schema.columns WHERE table_schema='cruft' AND table_name='temp';
"db","cruft","temp","id",1,"nextval('cruft.temp_id_seq'::regclass)","NO","bigint",,,64,2,0,,,,,,,,,,,,,"db","pg_catalog","int8",,,,,"1","NO","NO",,,,,,,"NEVER",,"YES"

データベース名(db)、スキーマ名(cruft)、テーブル名(temp)、列名(id)、デフォルト値(nextval(...))、データ型(bigintおよびint8 .. NOT bigserial) ...デフォルト値がシーケンスであるかどうかを確認するだけでよいことはわかっていますが、手動でシーケンスを作成し、デフォルト値が存在する非シリアル列を作成できるため、100%正確であるとは思いません。そのシーケンス。

誰か私がこれを達成する方法についての提案はありますか?nextval(* _ seq)のデフォルト値をチェックする以外に何かありますか?

TL; DRまたはpg_catalogに不慣れな新規ユーザーの場合にここに追加されたSQLソリューション用に編集:

with sequences as (
  select oid, relname as sequencename from pg_class where relkind = 'S'
) select
  sch.nspname as schemaname, tab.relname as tablename, col.attname as columnname, col.attnum as columnnumber, seqs.sequencename
from pg_attribute col
join pg_class tab on col.attrelid = tab.oid
join pg_namespace sch on tab.relnamespace = sch.oid
left join pg_attrdef def on tab.oid = def.adrelid and col.attnum = def.adnum
left join pg_depend deps on def.oid = deps.objid and deps.deptype = 'n'
left join sequences seqs on deps.refobjid = seqs.oid
where sch.nspname != 'information_schema' and sch.nspname not like 'pg_%' -- won't work if you have user schemas matching pg_
  and col.attnum > 0
  and seqs.sequencename is not null -- TO ONLY VIEW SERIAL/BIGSERIAL COLUMNS
order by sch.nspname, tab.relname, col.attnum;

1
コード例とその後、関連の答え:dba.stackexchange.com/questions/90555/...
アーウィンBrandstetter

回答:


8

SERIALBIGSERIALは一種の疑似タイプです。お気づきのとおり、これらは実際には内部的にはINTBIGINTにすぎません。

舞台裏で何が起こっているかは、PostgreSQLがシーケンスを作成し、それをテーブルへの依存関係に設定することです。シーケンス名とそれがテーブルにどのように関連しているかについてpg_classを検索できます。

pg_class:http ://www.postgresql.org/docs/9.2/static/catalog-pg-class.html

SQLフィドル:http ://sqlfiddle.com/#!12/ dfcbd/6

シーケンス関数:http : //www.postgresql.org/docs/9.2/static/functions-sequence.html

このStackOverflowの投稿が役立つ場合があります:https : //stackoverflow.com/questions/1493262/list-all-sequences-in-a-postgres-db-8-1-with-sql

UPDATEは:あなたが使用することもできpg_dependを:シーケンスはあなたが興味のあるテーブル/列に関連した把握するhttp://www.postgresql.org/docs/9.2/static/catalog-pg-depend.htmlを


10

ドキュメントが次のように述べているというエフェサーの答えに追加させてください:

smallserial、serial、bigserialのデータ型は真の型ではなく、一意の識別子列を作成するための表記上の便宜にすぎません(他のデータベースでサポートされているAUTO_INCREMENTプロパティと同様)。現在の実装では、以下を指定します:

CREATE TABLE tablename (
    colname SERIAL
);

以下を指定するのと同じです。

CREATE SEQUENCE tablename_colname_seq;
CREATE TABLE tablename (
    colname integer NOT NULL DEFAULT nextval('tablename_colname_seq')
);
ALTER SEQUENCE tablename_colname_seq OWNED BY tablename.colname;

つまり、

  • 列のデータ型が整数(bigint)であり、
  • NULLではない
  • デフォルト値は、問題の列が所有するシーケンスから取得されます

次に、それはserial列です。したがって、(を追加してNOT NULL)提案したように、カタログでこれらの要素を確認するだけで、serial列を特定できます。

(ビッグ)シリアルを見つけるための実際のクエリについては、Erwin Brandstetterの優れた回答を参照してください。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.