7

在 Postgres 8.4 或更高版本中,在不实际创建行的情况下获取默认填充的数据行的最有效方法是什么。例如,作为交易(伪代码):

create table "mytable"
(
  id serial PRIMARY KEY NOT NULL,
  parent_id integer NOT NULL DEFAULT 1,
  random_id integer NOT NULL DEFAULT random(),
)

begin transaction
  fake_row = insert into mytable (id) values (0) returning *;
  delete from mytable where id=0;
  return fake_row;
end transaction

基本上,我希望查询具有单行,其中 parent_id 为 1,random_id 是随机数(或其他函数返回值),但我不希望此记录保留在表中或影响主键序列 serial_id_seq。

我的选择似乎是使用上面的事务或创建视图,这些视图是添加了假行的表的副本,但我不知道每个的所有优点和缺点,或者是否存在更好的方法。

我正在寻找一个假设不知道任何列的数据类型或默认值的答案,除了列id的数量或顺序。只有表名是已知的,并且表中不应存在 id 为 0 的记录。

在过去,我创建了假记录 0 作为永久记录,但我开始将此记录视为一种污染(因为我通常必须将其从未来的查询中过滤掉)。

4

3 回答 3

6

您可以使用以下命令将表定义和默认值复制到临时表:

CREATE TEMP TABLE table_name_rt (LIKE table_name INCLUDING DEFAULTS);

并使用此临时表生成虚拟行。此类表将在会话(或事务)结束时被删除,并且仅对当前会话可见。

于 2013-08-01T08:17:47.650 回答
4

您可以查询目录并构建动态查询

假设我们有这张表:

create table test10(
      id serial primary key,
      first_name varchar( 100 ),
      last_name  varchar( 100 ) default 'Tom',
      age int not null default 38,
      salary float  default 100.22
);

当您运行以下查询时:

SELECT string_agg( txt, ' ' order by id ) 
FROM (
select 1 id, 'SELECT ' txt
union all
select 2, -9999 || ' as id '
union all
select 3, ', '  
       || coalesce( column_default,  'null'||'::'||c.data_type ) 
       || ' as ' || c.column_name
from information_schema.columns c
where table_schema = 'public'
    and table_name = 'test10'
    and ordinal_position > 1
) xx
    ;

结果你会得到这个刺痛:

"SELECT  -9999 as id  , null::character varying as first_name , 
'Tom'::character varying as last_name , 38 as age , 100.22 as salary"

然后执行此查询,您将获得“幻影行”。

我们可以构建一个函数来构建和执行查询并返回我们的行作为结果:

CREATE OR REPLACE FUNCTION get_phantom_rec (p_i test10.id%type ) 
returns test10 as $$
DECLARE 
    v_sql text;
    myrow test10%rowtype;
begin
   SELECT string_agg( txt, ' ' order by id ) 
   INTO v_sql
   FROM (
    select 1 id, 'SELECT ' txt
    union all
    select 2, p_i || ' as id '
    union all
    select 3, ', '  
           || coalesce( column_default,  'null'||'::'||c.data_type ) 
           || ' as ' || c.column_name
    from information_schema.columns c
    where table_schema = 'public'
        and table_name = 'test10'
        and ordinal_position > 1
    ) xx
    ;
    EXECUTE v_sql INTO myrow;
    RETURN  myrow;
END$$ LANGUAGE plpgsql ;

然后这个简单的查询给你你想要的:

select * from get_phantom_rec ( -9999 );

  id   | first_name | last_name | age | salary
-------+------------+-----------+-----+--------
 -9999 |            | Tom       |  38 | 100.22
于 2013-08-01T18:34:13.667 回答
2

我只会选择假值作为文字:

select 1 id, 1 parent_id, 1 user_id

返回的行将(实际上)与真实行无法区分。


要从目录中获取值:

select
  0 as id, -- special case for serial type, just return 0
  (select column_default::int -- Cast to int, because we know the column is int
   from INFORMATION_SCHEMA.COLUMNS
   where table_name = 'mytable'
   and column_name = 'parent_id') as parent_id,
  (select column_default::int -- Cast to int, because we know the column is int
   from INFORMATION_SCHEMA.COLUMNS
   where table_name = 'mytable'
   and column_name = 'user_id') as user_id;

请注意,您必须知道列是什么以及它们的类型,但这是合理的。如果您更改表架构(默认值除外),则需要调整查询。

将上述内容视为SQLFiddle

于 2013-08-01T04:33:33.503 回答