Как обновить из SELECT в SQL Server?

на SQL Server, возможно insert в таблице с помощью SELECT о себе:

INSERT INTO Table (col1, col2, col3)
SELECT col1, col2, col3 
FROM other_table 
WHERE sql = 'cool'

это обновление через SELECT? У меня есть временная таблица, содержащая значения, и я хотел бы обновить другую таблицу, используя эти значения. Возможно, что-то вроде этого:

UPDATE Table SET col1, col2
SELECT col1, col2 
FROM other_table 
WHERE sql = 'cool'
WHERE Table.id = other_table.id

29 ответов


UPDATE
    Table_A
SET
    Table_A.col1 = Table_B.col1,
    Table_A.col2 = Table_B.col2
FROM
    Some_Table AS Table_A
    INNER JOIN Other_Table AS Table_B
        ON Table_A.id = Table_B.id
WHERE
    Table_A.col3 = 'cool'

В SQL Server 2008 (или более), используйте MERGE

MERGE INTO YourTable T
   USING other_table S 
      ON T.id = S.id
         AND S.tsql = 'cool'
WHEN MATCHED THEN
   UPDATE 
      SET col1 = S.col1, 
          col2 = S.col2;

кроме того:

MERGE INTO YourTable T
   USING (
          SELECT id, col1, col2 
            FROM other_table 
           WHERE tsql = 'cool'
         ) S
      ON T.id = S.id
WHEN MATCHED THEN
   UPDATE 
      SET col1 = S.col1, 
          col2 = S.col2;

UPDATE table 
SET Col1 = i.Col1, 
    Col2 = i.Col2 
FROM (
    SELECT ID, Col1, Col2 
    FROM other_table) i
WHERE 
    i.ID = table.ID

Я бы изменить отличный ответ Робина следующим образом:

UPDATE Table
SET Table.col1 = other_table.col1,
 Table.col2 = other_table.col2
FROM
    Table
INNER JOIN other_table ON Table.id = other_table.id
WHERE
    Table.col1 != other_table.col1
OR Table.col2 != other_table.col2
OR (
    other_table.col1 IS NOT NULL
    AND Table.col1 IS NULL
)
OR (
    other_table.col2 IS NOT NULL
    AND Table.col2 IS NULL
)

без предложения WHERE вы повлияете даже на строки, которые не должны быть затронуты, что может (возможно) вызвать пересчет индекса или триггеры огня, которые действительно не должны были быть запущены.


в одну сторону

UPDATE t 
SET t.col1 = o.col1, 
    t.col2 = o.col2
FROM 
    other_table o 
  JOIN 
    t ON t.id = o.id
WHERE 
    o.sql = 'cool'

еще одна возможность, еще не упомянутая, - просто бросить SELECT сам оператор в CTE, а затем обновите CTE.

;WITH CTE
     AS (SELECT T1.Col1,
                T2.Col1 AS _Col1,
                T1.Col2,
                T2.Col2 AS _Col2
         FROM   T1
                JOIN T2
                  ON T1.id = T2.id
         /*Where clause added to exclude rows that are the same in both tables
           Handles NULL values correctly*/
         WHERE EXISTS(SELECT T1.Col1,
                             T1.Col2
                       EXCEPT
                       SELECT T2.Col1,
                              T2.Col2))
UPDATE CTE
SET    Col1 = _Col1,
       Col2 = _Col2

это имеет то преимущество, что его легко запустить SELECT оператор сам по себе сначала проверяет результаты, но для этого требуется псевдоним столбцов, как указано выше, если они называются одинаково в исходных и целевых таблицах.

это также имеет то же ограничение, что и proprietary UPDATE ... FROM синтаксис, показанный в четырех других ответы. Если исходная таблица находится на стороне "один ко многим", то неопределенно, какая из возможных совпадающих Соединенных записей будет использоваться в Update (вопрос MERGE избегает вызывает ошибку при попытке обновить одну и ту же строку более одного раза).


для записи (и другие ищут, как я был), вы можете сделать это в MySQL, как это:

UPDATE first_table, second_table
SET first_table.color = second_table.color
WHERE first_table.id = second_table.foreign_id

, используя псевдоним:

UPDATE t
   SET t.col1 = o.col1
  FROM table1 AS t
         INNER JOIN 
       table2 AS o 
         ON t.id = o.id

самый простой способ сделать это:

UPDATE
    table_to_update,
    table_info
SET
    table_to_update.col1 = table_info.col1,
    table_to_update.col2 = table_info.col2

WHERE
    table_to_update.ID = table_info.ID

Это может быть нишевой причиной для выполнения обновления (например, в основном используется в процедуре) или может быть очевидным для других, но следует также указать, что вы можете выполнить инструкцию update-select без использования join (в случае, если таблицы, которые вы обновляете, не имеют общего поля).

update
    Table
set
    Table.example = a.value
from
    TableExample a
where
    Table.field = *key value* -- finds the row in Table 
    AND a.field = *key value* -- finds the row in TableExample a

а вот еще один полезный синтаксис:

UPDATE suppliers
SET supplier_name = (SELECT customers.name
                     FROM customers
                     WHERE customers.customer_id = suppliers.supplier_id)
WHERE EXISTS (SELECT customers.name
              FROM customers
              WHERE customers.customer_id = suppliers.supplier_id);

он проверяет, является ли он нулевым или нет, используя "WHERE EXIST".


Я добавляю это только для того, чтобы вы могли увидеть быстрый способ написать его, чтобы вы могли проверить, что будет обновлено перед обновлением.

UPDATE Table 
SET  Table.col1 = other_table.col1,
     Table.col2 = other_table.col2 
--select Table.col1, other_table.col,Table.col2,other_table.col2, *   
FROM     Table 
INNER JOIN     other_table 
    ON     Table.id = other_table.id 

Если вы используете в MySQL вместо SQL Server используется синтаксис:

UPDATE Table1
INNER JOIN Table2
ON Table1.id = Table2.id
SET Table1.col1 = Table2.col1,
    Table1.col2 = Table2.col2

обновление от SELECT с внутренним соединением в базе данных SQL

поскольку слишком много ответов на этот пост, которые наиболее сильно проголосовали, я подумал, что дам свое предложение и здесь. Хотя вопрос очень интересный, я видел на многих сайтах форумов и сделал решение с помощью ВНУТРЕННЕЕ СОЕДИНЕНИЕ со скриншотами.

сначала я создал таблицу с именем schoolold и вставил несколько записей относительно их имена столбцов и выполнить его.

затем я выполнил выберите для просмотра вставленных записей.

enter image description here

затем я создал новую таблицу с именем schoolnew и аналогично выполненные выше действия.

enter image description here

затем, чтобы просмотреть вставленные в него записи, я выполняю команду SELECT.

enter image description here

Теперь Здесь Я хотите внести некоторые изменения в третью и четвертую строку, для завершения этого действия я выполняю обновление С ВНУТРЕННЕЕ СОЕДИНЕНИЕ.

enter image description here

для просмотра изменений я выполняю выберите.

enter image description here

вы можете видеть, как третья и четвертая записи таблицы schoolold легко заменяется таблице schoolnew используя внутреннее соединение с инструкция update.


и если вы хотите присоединиться к таблице с собой (что не будет происходить слишком часто):

update t1                    -- just reference table alias here
set t1.somevalue = t2.somevalue
from table1 t1               -- these rows will be the targets
inner join table1 t2         -- these rows will be used as source
on ..................        -- the join clause is whatever suits you

в следующем примере используется производная таблица, оператор SELECT после предложения FROM, для возврата старых и новых значений для дальнейших обновлений:

UPDATE x
SET    x.col1 = x.newCol1,
       x.col2 = x.newCol2
FROM   (SELECT t.col1,
               t2.col1 AS newCol1,
               t.col2,
               t2.col2 AS newCol2
        FROM   [table] t
               JOIN other_table t2
                 ON t.ID = t2.ID) x

Обновление через CTE более читаем, чем другие ответы здесь:

;WITH cte
     AS (SELECT col1,col2,id
         FROM   other_table
         WHERE  sql = 'cool')
UPDATE A
SET    A.col1 = B.col1,
       A.col2 = B.col2
FROM   table A
       INNER JOIN cte B
               ON A.id = B.id

Если вы используете SQL Server, вы можете обновить одну таблицу из другой без указания соединения и просто связать две из where предложения. Это упрощает SQL-запрос:

UPDATE Table1
SET Table1.col1 = Table2.col1,
    Table1.col2 = Table2.col2
FROM
    Table2
WHERE
    Table1.id = Table2.id

другой способ-использовать производную таблицу:

UPDATE t
SET t.col1 = a.col1
    ,t.col2 = a.col2
FROM (
SELECT id, col1, col2 FROM @tbl2) a
INNER JOIN @tbl1 t ON t.id = a.id

пример данных

DECLARE @tbl1 TABLE (id INT, col1 VARCHAR(10), col2 VARCHAR(10))
DECLARE @tbl2 TABLE (id INT, col1 VARCHAR(10), col2 VARCHAR(10))

INSERT @tbl1 SELECT 1, 'a', 'b' UNION SELECT 2, 'b', 'c'

INSERT @tbl2 SELECT 1, '1', '2' UNION SELECT 2, '3', '4'

UPDATE t
SET t.col1 = a.col1
    ,t.col2 = a.col2
FROM (
SELECT id, col1, col2 FROM @tbl2) a
INNER JOIN @tbl1 t ON t.id = a.id

SELECT * FROM @tbl1
SELECT * FROM @tbl2

UPDATE TQ
SET TQ.IsProcessed = 1, TQ.TextName = 'bla bla bla'
FROM TableQueue TQ
INNER JOIN TableComment TC ON TC.ID = TQ.TCID
WHERE TQ.IsProcessed = 0

чтобы убедиться, что вы обновляете то, что хотите, выберите first

SELECT TQ.IsProcessed, 1 AS NewValue1, TQ.TextName, 'bla bla bla' AS NewValue2
FROM TableQueue TQ
INNER JOIN TableComment TC ON TC.ID = TQ.TCID
WHERE TQ.IsProcessed = 0

использование:

drop table uno
drop table dos

create table uno
(
    uid int,
    col1 char(1),
    col2 char(2)
)
create table dos
(
    did int,
    col1 char(1),
    col2 char(2),
    [sql] char(4)
)
insert into uno(uid) values (1)
insert into uno(uid) values (2)
insert into dos values (1,'a','b',null)
insert into dos values (2,'c','d','cool')

select * from uno 
select * from dos

либо:

update uno set col1 = (select col1 from dos where uid = did and [sql]='cool'), 
col2 = (select col2 from dos where uid = did and [sql]='cool')

или:

update uno set col1=d.col1,col2=d.col2 from uno 
inner join dos d on uid=did where [sql]='cool'

select * from uno 
select * from dos

если имя столбца ID одинаково в обеих таблицах, просто поместите имя таблицы перед обновляемой таблицей и используйте псевдоним для выбранной таблицы, т. е.:

update uno set col1 = (select col1 from dos d where uno.[id] = d.[id] and [sql]='cool'),
col2  = (select col2 from dos d where uno.[id] = d.[id] and [sql]='cool')

ниже Решение работает для базы данных MySQL:

UPDATE table1 a , table2 b 
SET a.columname = 'some value' 
WHERE b.columnname IS NULL ;

существует даже более короткий метод, и это может быть удивительно для многих из вас:

-- Sample data:
---------------------------------------------------------------------------
CREATE TABLE #SOURCE ([ID] INT, [Desc] VARCHAR(10));
CREATE TABLE #DESTINATION ([ID] INT, [Desc] VARCHAR(10))

INSERT INTO #SOURCE VALUES(1,'Desc_1'), (2, 'Desc_2'), (3, 'Desc_3');
INSERT INTO #DESTINATION VALUES(1,'Desc_4'), (2, 'Desc_5'), (3, 'Desc_6');
---------------------------------------------------------------------------
UPDATE #DESTINATION
SET #DESTINATION.[Desc] = #SOURCE.[Desc]
FROM #SOURCE
WHERE #DESTINATION.[ID] = #SOURCE.[ID]
AND #Source.[Desc] = 'Desc_2'

в принятом ответе, после:

SET
Table_A.col1 = Table_B.col1,
Table_A.col2 = Table_B.col2

Я бы добавил:

OUTPUT deleted.*, inserted.*

то, что я обычно делаю, это положить все в рулонную транзакцию и использовать "OUTPUT": таким образом, я вижу все, что должно произойти. Когда я доволен тем, что вижу, я меняю ROLLBACK на COMMIT.

мне обычно нужно документировать то, что я сделал, поэтому я использую "results to Text" опция, когда я запускаю откатный запрос, и я сохраняю как сценарий, так и результат ВЫХОД. (Конечно, это не практично, если я изменил слишком много строк)


другой способ обновления из инструкции Select:

UPDATE A
SET A.col = A.col,B.col1 = B.col1
FROM  first_Table AS A
INNER JOIN second_Table AS B  ON A.id = B.id WHERE A.col2 = 'cool'

UPDATE table AS a
INNER JOIN table2 AS b
ON a.col1 = b.col1
INNER JOIN ... AS ...
ON ... = ...
SET ...
WHERE ...

консолидация всех подходы разные.

  1. выберите обновления
  2. обновление с общим табличным выражением
  3. слияние

структура таблицы образцов ниже и будет обновляться от Product_BAK до таблицы продуктов.

продукт

CREATE TABLE [dbo].[Product](
    [Id] [int] IDENTITY(1, 1) NOT NULL,
    [Name] [nvarchar](100) NOT NULL,
    [Description] [nvarchar](100) NULL
) ON [PRIMARY]

Product_BAK

    CREATE TABLE [dbo].[Product_BAK](
        [Id] [int] IDENTITY(1, 1) NOT NULL,
        [Name] [nvarchar](100) NOT NULL,
        [Description] [nvarchar](100) NULL
    ) ON [PRIMARY]

1. Выберите Обновить

    update P1
    set Name = P2.Name
    from Product P1
    inner join Product_Bak P2 on p1.id = P2.id
    where p1.id = 2

2. Обновление с общего стола выражение

    ; With CTE as
    (
        select id, name from Product_Bak where id = 2
    )
    update P
    set Name = P2.name
    from  product P  inner join CTE P2 on P.id = P2.id
    where P2.id = 2

3. Слияние

    Merge into product P1
    using Product_Bak P2 on P1.id = P2.id

    when matched then
    update set p1.[description] = p2.[description], p1.name = P2.Name;

в инструкции Merge мы можем сделать вставку, если не найдем соответствующую запись в цели, но существуем в источнике и, пожалуйста, найдите синтаксис:

    Merge into product P1
    using Product_Bak P2 on P1.id = P2.id

    when matched then
    update set p1.[description] = p2.[description], p1.name = P2.Name

    WHEN NOT MATCHED THEN
    insert (name, description)
    values(p2.name, P2.description);

вы можете использовать из этого для обновления в sql server

UPDATE
    T1
SET
   T1.col1 = T2.col1,
   T1.col2 = T2.col2
FROM
   Table1 AS T1
INNER JOIN Table2 AS T2
    ON T1.id = T2.id
WHERE
    T1.col3 = 'cool'

declare @tblStudent table (id int,name varchar(300))
declare @tblMarks table (std_id int,std_name varchar(300),subject varchar(50),marks int)

insert into @tblStudent Values (1,'Abdul')
insert into @tblStudent Values(2,'Rahim')

insert into @tblMarks Values(1,'','Math',50)
insert into @tblMarks Values(1,'','History',40)
insert into @tblMarks Values(2,'','Math',30)
insert into @tblMarks Values(2,'','history',80)


select * from @tblMarks

update m
set m.std_name=s.name
 from @tblMarks as m
left join @tblStudent as s on s.id=m.std_id

select * from @tblMarks