Запрос SQL для обновления родительской записи значениями дочерней записи - PullRequest
1 голос
/ 31 марта 2012

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

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

-- Create the test tables
CREATE TABLE projects (
  ProjectId varchar(16) PRIMARY KEY,
  ProjectName varchar(100),
  Codestring nvarchar(100)
)
GO
CREATE TABLE prcodes (
  CodeId varchar(16) PRIMARY KEY,
  Code varchar (4),
  ProjectId varchar(16)
)
GO
-- Add sample data to tables: Two projects records, one with 3 child records, the other with 2.
INSERT INTO projects
(ProjectId, ProjectName)
SELECT '101','Smith' UNION ALL
SELECT '102','Jones'
GO
INSERT INTO prcodes
(CodeId, Code, ProjectId)
SELECT 'A1','Blue', '101' UNION ALL
SELECT 'A2','Pink', '101' UNION ALL
SELECT 'A3','Gray', '101' UNION ALL
SELECT 'A4','Blue', '102' UNION ALL
SELECT 'A5','Gray', '102'
GO

Я застрял на том, как создать правильный запрос на обновление. Можете ли вы помочь исправить этот запрос?

-- Partially working, but stuffs all values, not just values from chile (prcodes) records of parent (projects)
UPDATE proj
SET
proj.Codestring = (SELECT STUFF((SELECT ',' + prc.Code 
FROM projects proj INNER JOIN prcodes prc ON proj.ProjectId = prc.ProjectId
ORDER BY 1 ASC FOR XML PATH('')),1, 1, ''))

Результат, который я получаю для поля Codestring в Projects:

    ProjectId   ProjectName Codestring
    101     Smith       Blue,Blue,Gray,Gray,Pink
    ...

Но результат, который мне нужен для поля Codestring в проектах:

    ProjectId   ProjectName Codestring
    101     Smith       Blue,Pink,Gray
    ...

Вот мой старт на триггере. Вышеупомянутый запрос на обновление будет добавлен к этому триггеру. Можете ли вы помочь мне выполнить запрос на создание триггера?

CREATE TRIGGER Update_Codestring ON prcodes
AFTER INSERT, UPDATE, DELETE
AS
WITH CTE AS (
  select ProjectId from inserted
  union
  select ProjectId from deleted
)

1 Ответ

2 голосов
/ 31 марта 2012

Следующий триггер будет работать так, как вы хотите.

CREATE TRIGGER Update_Codestring ON prcodes
AFTER INSERT, UPDATE, DELETE
AS
UPDATE projects
SET Codestring = (SELECT STUFF((SELECT ',' + prc.Code 
    FROM projects proj INNER JOIN prcodes prc ON proj.ProjectId = prc.ProjectId
    WHERE proj.ProjectId = projects.ProjectId
    ORDER BY 1 ASC FOR XML PATH('')),1, 1, ''))
where ProjectId in (SELECT ProjectId FROM inserted
                    UNION
                    SELECT ProjectId FROM deleted)

Чего вам не хватало в исходном операторе обновления:

WHERE proj.ProjectId = projects.ProjectId - это отфильтрует подзапрос только кпроект, который обновляется.projects без псевдонима поступает из оператора обновления, так что обновление применяется к каждой строке в projects только для текущей строки проекта, которая обновляется.

WHERE ProjectId IN (SELECT ProjectId FROM inserted UNION SELECT ProjectId FROM deleted) - Это отфильтрует обновление, чтобы повлиять только настроки с измененными дочерними элементами.

Также вы можете упростить оператор обновления, поскольку для него не требуется дважды включать таблицу projects:

CREATE TRIGGER Update_Codestring ON prcodes
AFTER INSERT, UPDATE, DELETE
AS
UPDATE projects
SET Codestring = (SELECT STUFF((SELECT ',' + prc.Code 
    FROM prcodes prc
    WHERE prc.ProjectId = projects.ProjectId
    ORDER BY 1 ASC FOR XML PATH('')),1, 1, ''))
WHERE ProjectId IN (SELECT ProjectId FROM inserted
                    UNION
                    SELECT ProjectId FROM deleted)

Наконец, вам действительно нужно хранить Codestring на вашем Projects столе?Это то, что можно легко пересчитать в запросе в любое время или даже представить в виде.Это значит, что вам не нужно беспокоиться о необходимости хранить дополнительные данные и иметь триггер для их сохранения.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...