Insert into ... values ( SELECT ... FROM ... )
Asked Answered
T

27

1840

I am trying to INSERT INTO a table using the input from another table. Although this is entirely feasible for many database engines, I always seem to struggle to remember the correct syntax for the SQL engine of the day (MySQL, Oracle, SQL Server, Informix, and DB2).

Is there a silver-bullet syntax coming from an SQL standard (for example, SQL-92) that would allow me to insert the values without worrying about the underlying database?

Terracotta answered 25/8, 2008 at 12:45 Comment(2)
this example works: insert into tag_zone select @tag,zoneid,GETDATE(),@positiong.STIntersects(polygon) from zoneLevorotation
See also Combining INSERT INTO and WITH/CTE.Sybille
T
2024

Try:

INSERT INTO table1 ( column1 )
SELECT  col1
FROM    table2  

This is standard ANSI SQL and should work on any DBMS

It definitely works for:

  • Oracle
  • MS SQL Server
  • MySQL
  • Postgres
  • SQLite v3
  • Teradata
  • DB2
  • Sybase
  • Vertica
  • HSQLDB
  • H2
  • AWS RedShift
  • SAP HANA
  • Google Spanner
Terracotta answered 25/8, 2008 at 12:47 Comment(3)
I highly doubt all of them supported it right away - only SQLite has a minimum version, but it would also be interesting for others, especially ORACLE.Arliearliene
@MindRoasterMir You can't do that. If you perform an insert you will create a new row in the table hence you cannot have an existing row with which you can compare (c.pageno = t.pageno). You are probably looking for an UPDATE or MERGE (MS Sql)Bloodthirsty
IF you want to handle duplicate row according to primary key or unique index, you can use IGNORE option like INSERT IGNORE INTO table SELECT * FROM another_tableDillard
T
1237

Claude Houle's answer: should work fine, and you can also have multiple columns and other data as well:

INSERT INTO table1 ( column1, column2, someInt, someVarChar )
SELECT  table2.column1, table2.column2, 8, 'some string etc.'
FROM    table2
WHERE   table2.ID = 7;

I've only used this syntax with Access, SQL 2000/2005/Express, MySQL, and PostgreSQL, so those should be covered. It should also work with SQLite3.

Tighten answered 25/8, 2008 at 14:11 Comment(4)
what if, the where condition changed to table2.country and returns number of rows greater than one? I got similar issue here: #36030870Extant
There shouldn't be a problem with inserting more than one row.Unpolitic
is it Necessary that we insert into all columns of the tablePita
@maheshmnj no, only the columns that are set to NOT NULL and no default value need to be included, any other columns will be set to their default values or NULLTighten
M
234

To get only one value in a multi value INSERT from another table I did the following in SQLite3:

INSERT INTO column_1 ( val_1, val_from_other_table ) 
VALUES('val_1', (SELECT  val_2 FROM table_2 WHERE val_2 = something))
Mediator answered 10/1, 2014 at 23:46 Comment(5)
Just for clarification: this is incorrect for SQLite3. As per the documentation, the source data for the INSERT is either VALUES or a SELECT statement, not both.Acropolis
It is true the documentation does not list it, but it does work. Regardless, I think that using the select statement instead of values does make it more readable.Kirman
It works for specifying a value inside a row, but the more general case requires getting lots of rows.Pallet
If val_1 doesn't change across rows, then the following syntax might work in SQLite3? select 'foo', some_column from some_table - works in SQLServer 2014Raguelragweed
The documentation does list this (now?): this syntax is INSERT INTO ... VALUES ([expr], [expr], ...) and one of the paths in [expr] is {{NOT} EXISTS} ([select-stmt]) - note that the paranthesis around the select statement are required ({} meaning optional)Rodneyrodolfo
D
75

Both the answers I see work fine in Informix specifically, and are basically standard SQL. That is, the notation:

INSERT INTO target_table[(<column-list>)] SELECT ... FROM ...;

works fine with Informix and, I would expect, all the DBMS. (Once upon 5 or more years ago, this is the sort of thing that MySQL did not always support; it now has decent support for this sort of standard SQL syntax and, AFAIK, it would work OK on this notation.) The column list is optional but indicates the target columns in sequence, so the first column of the result of the SELECT will go into the first listed column, etc. In the absence of the column list, the first column of the result of the SELECT goes into the first column of the target table.

What can be different between systems is the notation used to identify tables in different databases - the standard has nothing to say about inter-database (let alone inter-DBMS) operations. With Informix, you can use the following notation to identify a table:

[dbase[@server]:][owner.]table

That is, you may specify a database, optionally identifying the server that hosts that database if it is not in the current server, followed by an optional owner, dot, and finally the actual table name. The SQL standard uses the term schema for what Informix calls the owner. Thus, in Informix, any of the following notations could identify a table:

table
"owner".table
dbase:table
dbase:owner.table
dbase@server:table
dbase@server:owner.table

The owner in general does not need to be quoted; however, if you do use quotes, you need to get the owner name spelled correctly - it becomes case-sensitive. That is:

someone.table
"someone".table
SOMEONE.table

all identify the same table. With Informix, there's a mild complication with MODE ANSI databases, where owner names are generally converted to upper-case (informix is the exception). That is, in a MODE ANSI database (not commonly used), you could write:

CREATE TABLE someone.table ( ... )

and the owner name in the system catalog would be "SOMEONE", rather than 'someone'. If you enclose the owner name in double quotes, it acts like a delimited identifier. With standard SQL, delimited identifiers can be used many places. With Informix, you can use them only around owner names -- in other contexts, Informix treats both single-quoted and double-quoted strings as strings, rather than separating single-quoted strings as strings and double-quoted strings as delimited identifiers. (Of course, just for completeness, there is an environment variable, DELIMIDENT, that can be set - to any value, but Y is safest - to indicate that double quotes always surround delimited identifiers and single quotes always surround strings.)

Note that MS SQL Server manages to use [delimited identifiers] enclosed in square brackets. It looks weird to me, and is certainly not part of the SQL standard.

Dendrology answered 28/9, 2008 at 3:18 Comment(0)
A
69

Two approaches for insert into with select sub-query.

  1. With SELECT subquery returning results with One row.
  2. With SELECT subquery returning results with Multiple rows.

1. Approach for With SELECT subquery returning results with one row.

INSERT INTO <table_name> (<field1>, <field2>, <field3>) 
VALUES ('DUMMY1', (SELECT <field> FROM <table_name> ),'DUMMY2');

In this case, it assumes SELECT Sub-query returns only one row of result based on WHERE condition or SQL aggregate functions like SUM, MAX, AVG etc. Otherwise it will throw error

2. Approach for With SELECT subquery returning results with multiple rows.

INSERT INTO <table_name> (<field1>, <field2>, <field3>) 
SELECT 'DUMMY1', <field>, 'DUMMY2' FROM <table_name>;

The second approach will work for both the cases.

Atop answered 4/4, 2019 at 4:9 Comment(4)
i want to use RETURNING with this inorder to return a specific column of the inserted records ? how do i accomplish that @Mohammed Safeer any inputs appreciated ?Factory
@SithijaPiyumanThewaHettige you can do something like: INSERT INTO tablename (col1, col2) SELECT field1, field2 FROM tablename RETURNING col1, col2 The maria db support is from version 10.5.0 You can look at the documentation mariadb.com/kb/en/insertreturningSouthing
1) works with one row or less. (NULL will be inserted if zero rows.)Nationality
BTW, note that tables have columns, not fields.Nationality
L
50

To add something in the first answer, when we want only few records from another table (in this example only one):

INSERT INTO TABLE1
(COLUMN1, COLUMN2, COLUMN3, COLUMN4) 
VALUES (value1, value2, 
(SELECT COLUMN_TABLE2 
FROM TABLE2
WHERE COLUMN_TABLE2 like "blabla"),
value4);
Landwehr answered 9/4, 2015 at 17:15 Comment(1)
This approach only applies to such subquery that only one column is selected. In the case of multiple-column subquery, an error 'subquery must return only one column' will be raised. Adopt @travis's answer then.Allegraallegretto
L
39

Instead of VALUES part of INSERT query, just use SELECT query as below.

INSERT INTO table1 ( column1 , 2, 3... )
SELECT col1, 2, 3... FROM table2
Lachlan answered 21/4, 2015 at 10:31 Comment(0)
T
37

Most of the databases follow the basic syntax,

INSERT INTO TABLE_NAME
SELECT COL1, COL2 ...
FROM TABLE_YOU_NEED_TO_TAKE_FROM
;

Every database I have used follow this syntax namely, DB2, SQL Server, MY SQL, PostgresQL

Tatia answered 1/4, 2013 at 10:9 Comment(0)
D
32

This can be done without specifying the columns in the INSERT INTO part if you are supplying values for all columns in the SELECT part.

Let's say table1 has two columns. This query should work:

INSERT INTO table1
SELECT  col1, col2
FROM    table2

This WOULD NOT work (value for col2 is not specified):

INSERT INTO table1
SELECT  col1
FROM    table2

I'm using MS SQL Server. I don't know how other RDMS work.

Defecate answered 16/10, 2012 at 14:19 Comment(0)
I
26

Just use parenthesis for SELECT clause into INSERT. For example like this :

INSERT INTO Table1 (col1, col2, your_desired_value_from_select_clause, col3)
VALUES (
   'col1_value', 
   'col2_value',
   (SELECT col_Table2 FROM Table2 WHERE IdTable2 = 'your_satisfied_value_for_col_Table2_selected'),
   'col3_value'
);
Inca answered 24/9, 2018 at 9:25 Comment(0)
H
25

This is another example using values with select:

INSERT INTO table1(desc, id, email) 
SELECT "Hello World", 3, email FROM table2 WHERE ...
Haem answered 20/3, 2014 at 9:12 Comment(0)
S
21

Simple insertion when table column sequence is known:

    Insert into Table1
    values(1,2,...)

Simple insertion mentioning column:

    Insert into Table1(col2,col4)
    values(1,2)

Bulk insertion when number of selected columns of a table(#table2) are equal to insertion table(Table1)

    Insert into Table1 {Column sequence}
    Select * -- column sequence should be same.
       from #table2

Bulk insertion when you want to insert only into desired column of a table(table1):

    Insert into Table1 (Column1,Column2 ....Desired Column from Table1)  
    Select Column1,Column2..desired column from #table2
       from #table2
Sokoto answered 13/2, 2014 at 12:23 Comment(0)
M
17

Here is another example where source is taken using more than one table:

INSERT INTO cesc_pf_stmt_ext_wrk( 
  PF_EMP_CODE    ,
  PF_DEPT_CODE   ,
  PF_SEC_CODE    ,
  PF_PROL_NO     ,
  PF_FM_SEQ      ,
  PF_SEQ_NO      ,
  PF_SEP_TAG     ,
  PF_SOURCE) 
SELECT
  PFl_EMP_CODE    ,
  PFl_DEPT_CODE   ,
  PFl_SEC         ,
  PFl_PROL_NO     ,
  PF_FM_SEQ       ,
  PF_SEQ_NO       ,
  PFl_SEP_TAG     ,
  PF_SOURCE
 FROM cesc_pf_stmt_ext,
      cesc_pfl_emp_master
 WHERE pfl_sep_tag LIKE '0'
   AND pfl_emp_code=pf_emp_code(+);

COMMIT;
Miceli answered 6/6, 2012 at 12:41 Comment(0)
O
17

Here's how to insert from multiple tables. This particular example is where you have a mapping table in a many to many scenario:

insert into StudentCourseMap (StudentId, CourseId) 
SELECT  Student.Id, Course.Id FROM Student, Course 
WHERE Student.Name = 'Paddy Murphy' AND Course.Name = 'Basket weaving for beginners'

(I realise matching on the student name might return more than one value but you get the idea. Matching on something other than an Id is necessary when the Id is an Identity column and is unknown.)

Odine answered 23/3, 2016 at 14:31 Comment(0)
S
15

You could try this if you want to insert all column using SELECT * INTO table.

SELECT  *
INTO    Table2
FROM    Table1;
Soldierly answered 17/6, 2016 at 10:28 Comment(2)
Not supported by H2Buckboard
Neither ANSI SQL nor database agnostic. INTO ruins it.Nationality
F
14

I actually prefer the following in SQL Server 2008:

SELECT Table1.Column1, Table1.Column2, Table2.Column1, Table2.Column2, 'Some String' AS SomeString, 8 AS SomeInt
INTO Table3
FROM Table1 INNER JOIN Table2 ON Table1.Column1 = Table2.Column3

It eliminates the step of adding the Insert () set, and you just select which values go in the table.

Finagle answered 22/3, 2013 at 14:57 Comment(0)
E
14

This worked for me:

insert into table1 select * from table2

The sentence is a bit different from Oracle's.

Ejaculation answered 20/11, 2013 at 12:19 Comment(0)
I
14
INSERT INTO yourtable
SELECT fielda, fieldb, fieldc
FROM donortable;

This works on all DBMS

Illuminative answered 20/5, 2015 at 8:44 Comment(0)
M
13

For Microsoft SQL Server, I will recommend learning to interpret the SYNTAX provided on MSDN. With Google it's easier than ever, to look for syntax.

For this particular case, try

Google: insert site:microsoft.com

The first result will be http://msdn.microsoft.com/en-us/library/ms174335.aspx

scroll down to the example ("Using the SELECT and EXECUTE options to insert data from other tables") if you find it difficult to interpret the syntax given at the top of the page.

[ WITH <common_table_expression> [ ,...n ] ]
INSERT 
{
        [ TOP ( expression ) [ PERCENT ] ] 
        [ INTO ] 
        { <object> | rowset_function_limited 
          [ WITH ( <Table_Hint_Limited> [ ...n ] ) ]
        }
    {
        [ ( column_list ) ] 
        [ <OUTPUT Clause> ]
        { VALUES ( { DEFAULT | NULL | expression } [ ,...n ] ) [ ,...n     ] 
        | derived_table       <<<<------- Look here ------------------------
        | execute_statement   <<<<------- Look here ------------------------
        | <dml_table_source>  <<<<------- Look here ------------------------
        | DEFAULT VALUES 
        }
    }
}
[;]

This should be applicable for any other RDBMS available there. There is no point in remembering all the syntax for all products IMO.

Morton answered 17/10, 2012 at 16:56 Comment(3)
I completely disagree, I've been looking at those syntax statements for years and still can't make heads or tails of them. Examples are much more usefulThermodynamics
This isn't an answer, it's saying "read the docs" and that's about itThermodynamics
Neither ANSI SQL nor database agnostic. TOP ruins it.Nationality
C
13
INSERT INTO FIRST_TABLE_NAME (COLUMN_NAME)
SELECT  COLUMN_NAME
FROM    ANOTHER_TABLE_NAME 
WHERE CONDITION;
Corney answered 13/2, 2018 at 16:35 Comment(0)
T
12

Best way to insert multiple records from any other tables.

INSERT  INTO dbo.Users
            ( UserID ,
              Full_Name ,
              Login_Name ,
              Password
            )
            SELECT  UserID ,
                    Full_Name ,
                    Login_Name ,
                    Password
            FROM    Users_Table
            (INNER JOIN / LEFT JOIN ...)
            (WHERE CONDITION...)
            (OTHER CLAUSE)
Telepathist answered 7/6, 2018 at 6:59 Comment(0)
L
10
select *
into tmp
from orders

Looks nice, but works only if tmp doesn't exists (creates it and fills). (SQL sever)

To insert into existing tmp table:

set identity_insert tmp on

insert tmp 
([OrderID]
      ,[CustomerID]
      ,[EmployeeID]
      ,[OrderDate]
      ,[RequiredDate]
      ,[ShippedDate]
      ,[ShipVia]
      ,[Freight]
      ,[ShipName]
      ,[ShipAddress]
      ,[ShipCity]
      ,[ShipRegion]
      ,[ShipPostalCode]
      ,[ShipCountry] )
      select * from orders

set identity_insert tmp off
Libertylibia answered 17/5, 2014 at 20:28 Comment(0)
J
8

IF you want to insert some data into a table without want to write column name.

INSERT INTO CUSTOMER_INFO
   (SELECT CUSTOMER_NAME,
           MOBILE_NO,
           ADDRESS
      FROM OWNER_INFO cm)

Where the tables are:

            CUSTOMER_INFO               ||            OWNER_INFO
----------------------------------------||-------------------------------------
CUSTOMER_NAME | MOBILE_NO | ADDRESS     || CUSTOMER_NAME | MOBILE_NO | ADDRESS 
--------------|-----------|---------    || --------------|-----------|--------- 
      A       |     +1    |   DC        ||       B       |     +55   |   RR  

Result:

            CUSTOMER_INFO               ||            OWNER_INFO
----------------------------------------||-------------------------------------
CUSTOMER_NAME | MOBILE_NO | ADDRESS     || CUSTOMER_NAME | MOBILE_NO | ADDRESS 
--------------|-----------|---------    || --------------|-----------|--------- 
      A       |     +1    |   DC        ||       B       |     +55   |   RR
      B       |     +55   |   RR        ||
Jerad answered 1/6, 2020 at 15:11 Comment(0)
P
5

If you go the INSERT VALUES route to insert multiple rows, make sure to delimit the VALUES into sets using parentheses, so:

INSERT INTO `receiving_table`
  (id,
  first_name,
  last_name)
VALUES 
  (1002,'Charles','Babbage'),
  (1003,'George', 'Boole'),
  (1001,'Donald','Chamberlin'),
  (1004,'Alan','Turing'),
  (1005,'My','Widenius');

Otherwise MySQL objects that "Column count doesn't match value count at row 1", and you end up writing a trivial post when you finally figure out what to do about it.

Peay answered 9/6, 2017 at 3:51 Comment(2)
The question is "insert into a table using the input from another table". How does your answer address this question?Educator
Eh dont be too hard on him. It answered my question when I was googling around. @QualityCatalystLeifeste
S
3

If you create table firstly you can use like this;

  select * INTO TableYedek From Table

This metot insert values but differently with creating new copy table.

Saltworks answered 13/7, 2021 at 15:27 Comment(2)
If you save whole table as backup use this query..select * INTO TableYedek_Backup From TableYedekPatronage
Neither ANSI SQL nor database agnostic. INTO ruins it.Nationality
C
1

In informix it works as Claude said:

INSERT INTO table (column1, column2) 
VALUES (value1, value2);    
Cima answered 12/12, 2019 at 7:59 Comment(0)
J
1

Postgres supports next: create table company.monitor2 as select * from company.monitor;

Jochebed answered 1/4, 2020 at 13:48 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.