postgresql: INSERT INTO ... (ВЫБРАТЬ * ...) - PullRequest
104 голосов
/ 21 мая 2011

Я не уверен, что это стандартный SQL:

 INSERT INTO tblA 
 (SELECT id, time 
    FROM tblB 
   WHERE time > 1000)  

Что я ищу: что если tblA и tblB находятся на разных серверах БД .

Предоставляет ли PostgreSql какую-либо утилиту или обладает какой-либо функциональностью, которая поможет использовать INSERT query with PGresult struct

Я имею в виду SELECT id, time FROM tblB ... вернет PGresult* при использовании PQexec. Можно ли использовать эту структуру в другом PQexec для выполнения команды INSERT.

EDIT:
Если это невозможно, я бы взял извлечение значений из PQresult * и создал бы синтаксис нескольких операторов INSERT, например:

INSERT INTO films (code, title, did, date_prod, kind) VALUES
    ('B6717', 'Tampopo', 110, '1985-02-10', 'Comedy'),
    ('HG120', 'The Dinner Game', 140, DEFAULT, 'Comedy'); 

Можно ли из этого создать готовое утверждение !! (

Ответы [ 6 ]

128 голосов
/ 21 мая 2011

Как писал Хенрик, вы можете использовать dblink для подключения к удаленной базе данных и получения результатов.Например:

psql dbtest
CREATE TABLE tblB (id serial, time integer);
INSERT INTO tblB (time) VALUES (5000), (2000);

psql postgres
CREATE TABLE tblA (id serial, time integer);

INSERT INTO tblA
    SELECT id, time 
    FROM dblink('dbname=dbtest', 'SELECT id, time FROM tblB')
    AS t(id integer, time integer)
    WHERE time > 1000;

TABLE tblA;
 id | time 
----+------
  1 | 5000
  2 | 2000
(2 rows)

PostgreSQL имеет record псевдотип (только для аргумента функции или типа результата), что позволяет запрашивать данные из другой (неизвестной) таблицы.

Редактировать:

Вы можете сделать это как подготовленное утверждение, если хотите, и оно также работает:

PREPARE migrate_data (integer) AS
INSERT INTO tblA
    SELECT id, time
    FROM dblink('dbname=dbtest', 'SELECT id, time FROM tblB')
    AS t(id integer, time integer)
    WHERE time > $1;

EXECUTE migrate_data(1000);
-- DEALLOCATE migrate_data;

Редактировать (да, другое):

Я простоувидел ваш пересмотренный вопрос (закрыт как дубликат или просто очень похож на это).

Если мое понимание верно (postgres имеет tbla, а dbtest имеет tblb, и вы хотите удаленная вставкас локальным выбором , а не дистанционный выбор с локальной вставкой , как указано выше):

psql dbtest

SELECT dblink_exec
(
    'dbname=postgres',
    'INSERT INTO tbla
        SELECT id, time
        FROM dblink
        (
            ''dbname=dbtest'',
            ''SELECT id, time FROM tblb''
        )
        AS t(id integer, time integer)
        WHERE time > 1000;'
);

Мне не нравится эта вложенная dblink, но AFAIK Я не могу ссылаться на tblBв теле dblink_exec .Используйте LIMIT для указания 20 верхних строк, но я думаю, что вам нужно сначала отсортировать их, используя предложение ORDER BY.

23 голосов
/ 02 февраля 2015

Если вы хотите вставить в указанный столбец:

INSERT INTO table (time)
(SELECT time FROM 
    dblink('dbname=dbtest', 'SELECT time FROM tblB') AS t(time integer) 
    WHERE time > 1000
);
9 голосов
/ 21 мая 2011

Вы можете использовать dblink , чтобы создать представление, разрешенное в другой базе данных.Эта база данных может находиться на другом сервере.

8 голосов
/ 07 февраля 2018

Эта запись (впервые увиденная здесь ) также выглядит полезной:

insert into postagem (
  resumopostagem,
  textopostagem,
  dtliberacaopostagem,
  idmediaimgpostagem,
  idcatolico,
  idminisermao,
  idtipopostagem
) select
  resumominisermao,
  textominisermao,
  diaminisermao,
  idmediaimgminisermao,
  idcatolico ,
  idminisermao,
  1
from
  minisermao    
1 голос
/ 26 июля 2018

Вот альтернативное решение, без использования dblink.

Предположим, B представляет исходную базу данных, а A представляет целевую базу данных: Тогда,

  1. Копировать таблицу из исходной БД в целевую БД:

    pg_dump -t <source_table> <source_db> | psql <target_db>
    
  2. Откройте приглашение psql, подключитесь к target_db и используйте простое insert:

    psql
    # \c <target_db>;
    # INSERT INTO <target_table>(id, x, y) SELECT id, x, y FROM <source_table>;
    
  3. В конце удалите копию source_table , которую вы создали в target_table .

    # DROP TABLE <source_table>;
    
0 голосов
/ 10 декабря 2018
insert into TABLENAMEA (A,B,C,D) 
select A::integer,B,C,D from TABLENAMEB
...