Внешнее соединение T-SQL не работает - PullRequest
0 голосов
/ 17 июля 2010

У меня есть запрос, который я написал с помощью JOIN для отображения сводки с использованием серийного номера детали.

Очевидно, это не работает!

Используя прямой поиск здесь:

select Serial_Number, Op_ID, Date_Time as 'DecayDate', 
    DECAY.System_ID AS 'DecayID', Test_Result as 'DecayResult' 
from Test_Results
where serial_number='CP21295 1006 09' and system_id like '%decay%'

Я получаю этот пример вывода:

Serial_Number     Op_ID   DecayDate               DecayID      DecayResult
CP21295 1006 09   003468  2009-10-15 06:36:13.643 AA_DECAY_6   FAILED
CP21295 1006 09   003468  2009-10-05 08:08:38.503 AA_DECAY_6   PASSED

Обратите внимание, что для этого одного номера детали были проведены два (2) теста.

Теперь к созданному мной запросу, который не работает:


Эта первая версия вернет ONE запись:

SELECT LABEL.Serial_Number, 
  DECAY.Op_ID,
  DECAY.Date_Time AS 'DecayDate',
  DECAY.System_ID AS 'DecayID',
  CASE
    WHEN DECAY.Test_Result LIKE '%fail%' THEN 'FAIL'
    WHEN DECAY.Test_Result LIKE '%pass%' THEN 'PASS'
    ELSE NULL
  END AS 'DecayResult'
  FROM 
  ACP_Parts AS LABEL
  LEFT OUTER JOIN (
    SELECT I.Serial_Number, I.Op_ID, I.Date_Time, I.System_ID, I.Test_Result
        FROM Test_Results I
    LEFT OUTER JOIN Test_Results O ON (
        I.Serial_Number=O.Serial_Number) AND 
        (O.Date_Time<I.Date_Time) -- AND (O.Serial_Number IS NULL) // didn't work
    WHERE
        (I.Test_Result LIKE '%pass%' OR I.Test_Result LIKE '%fail%') AND 
        (I.System_ID like '%decay%') AND
        (O.Test_Result LIKE '%pass%' OR O.Test_Result LIKE '%fail%') AND
        (O.System_ID like '%decay%')
    ) AS DECAY ON 
        (LABEL.Serial_Number=DECAY.Serial_Number) AND 
        (LABEL.Date_Time<DECAY.Date_Time)
  WHERE 
    (LABEL.Serial_Number IN (
        SELECT DISTINCT Serial_Number 
        FROM ACP_Parts 
        WHERE (Serial_Number IS NOT NULL) AND 
         (DATEADD(yy, - 1, GETDATE()) < Date_Time)))

выход

Serial_Number     DecayOp DecayDate               DecayID              DecayResult
CP21295 1006 09   003468  2009-10-15 06:36:13.643 AA_DECAY_6           PASS 

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


Я изменил запрос выше, надеясь вернуть самую новую запись, но эта версия ничего не возвращает:

SELECT LABEL.Serial_Number, 
  DECAY.Op_ID,
  DECAY.Date_Time AS 'DecayDate',
  DECAY.System_ID 'DecayID',
  CASE
    WHEN DECAY.Test_Result LIKE '%fail%' THEN 'FAIL'
    WHEN DECAY.Test_Result LIKE '%pass%' THEN 'PASS'
    ELSE NULL
  END AS 'DecayResult'
  FROM 
  ACP_Parts AS LABEL
  LEFT OUTER JOIN (
    SELECT TOP 1 Serial_Number, Op_ID, Date_Time, System_ID, Test_Result
    FROM Test_Results
    WHERE
      (System_ID like '%decay%') AND
      (Test_Result LIKE '%pass%' OR Test_Result LIKE '%fail%')
    ORDER BY Date_Time DESC
    ) AS DECAY ON 
        (LABEL.Serial_Number=DECAY.Serial_Number) AND 
        (LABEL.Date_Time<DECAY.Date_Time)
  WHERE 
    (LABEL.Serial_Number IN (
        SELECT DISTINCT Serial_Number 
        FROM ACP_Parts 
        WHERE (Serial_Number IS NOT NULL) AND 
         (DATEADD(yy, - 1, GETDATE()) < Date_Time)))

выход

Serial_Number     DecayOp DecayDate               DecayID              DecayResult
CP21295 1006 09   NULL    NULL                    NULL                 NULL

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

Ответы [ 2 ]

0 голосов
/ 20 июля 2010

ОК, я нашел способы настроить код для получения желаемых результатов.

Сначала, вот данные, которые я использовал:

CREATE TABLE JP1 (Serial_Number VarChar(20), Date_Time DateTime, Test_Result VarChar(255))
INSERT JP1 VALUES ('T900 09', '2009-10-10 01:15:00.000', 'NO TEST, Failed to Dump')
INSERT JP1 VALUES ('T901 09', '2009-10-10 01:20:00.000', '(BLUE) PASS (WHITE) FAIL')

CREATE TABLE JP2 (Serial_Number VarChar(20), Date_Time DateTime, Test_Result VarChar(255))
INSERT JP2 VALUES ('T900 09', '2009-10-15 01:15:00.000', 'NO TEST, Failed to Dump')
INSERT JP2 VALUES ('T900 09', '2009-10-15 02:15:00.000', '(BLUE) PASS (WHITE) FAIL')
INSERT JP2 VALUES ('T900 09', '2009-10-15 04:25:00.000', 'TEST ABORTED')
INSERT JP2 VALUES ('T900 09', '2009-10-15 04:59:00.000', '(BLUE) FAIL (WHITE) FAIL')
INSERT JP2 VALUES ('T900 09', '2009-10-15 05:09:00.000', '(WHITE) NO TEST')
INSERT JP2 VALUES ('T900 09', '2009-10-15 05:34:00.000', 'B=Pass.,W=FAIL, Final Fill')
INSERT JP2 VALUES ('T900 09', '2009-10-15 06:06:00.000', 'PASSED PD')
INSERT JP2 VALUES ('T900 09', '2009-10-15 06:30:00.000', 'TEST ABORTED')
INSERT JP2 VALUES ('T900 09', '2009-10-15 06:36:00.000', 'PASS')
INSERT JP2 VALUES ('T900 09', '2009-10-15 06:50:00.000', 'FAIL')
INSERT JP2 VALUES ('T900 09', '2009-10-05 08:08:00.000', 'NO TEST, Operator cancel')

Далее я создал два (2) очень похожие запросы.Оказывается, требования, изложенные в предложении LEFT OUTER JOIN, должны быть имитированы в предложении WHERE.

Вот как я извлекаю результат и дату FIRST PASS:

SELECT LABEL.Serial_Number, 
    DECAY.Date_Time AS 'DecayDate',
    CASE
    WHEN DECAY.Test_Result LIKE '%fail%' THEN 'FAIL'
    WHEN DECAY.Test_Result LIKE '%pass%' THEN 'PASS'
    ELSE NULL
    END AS 'DecayResult'
FROM JP1 AS LABEL
LEFT OUTER JOIN (
    SELECT I.Serial_Number, I.Date_Time, I.Test_Result
    FROM
        JP2 AS I
        LEFT OUTER JOIN JP2 AS O
        ON
        (I.Serial_Number=O.Serial_Number) AND
        ((O.Test_Result LIKE '%pass%') OR (O.Test_Result LIKE '%fail%')) AND -- //Gets First Test!
        (O.Test_Result NOT LIKE '%NO TEST%') AND
        (O.Date_Time<I.Date_Time) -- //(selects FIRST pass or fail)
        -- (I.Date_Time<O.Date_Time) -- //(selects LAST pass or fail)
    WHERE
        (I.Test_Result LIKE '%pass%' OR I.Test_Result LIKE '%fail%') AND -- //Gets First Test!
        (I.Test_Result NOT LIKE '%NO TEST%') AND
        (O.Serial_Number IS NULL)
) AS DECAY ON 
(LABEL.Serial_Number=DECAY.Serial_Number) AND 
(LABEL.Date_Time<DECAY.Date_Time)
WHERE 
(LABEL.Serial_Number IN (
    SELECT DISTINCT Serial_Number 
    FROM JP1
    WHERE (Serial_Number IS NOT NULL) AND 
    (DATEADD(yy, - 1, GETDATE()) < Date_Time))
)

Вот как я извлекаю FIRST Результат теста и дату:

SELECT LABEL.Serial_Number, 
    DECAY.Date_Time AS 'DecayDate',
    CASE
    WHEN DECAY.Test_Result LIKE '%fail%' THEN 'FAIL'
    WHEN DECAY.Test_Result LIKE '%pass%' THEN 'PASS'
    ELSE NULL
    END AS 'DecayResult'
FROM JP1 AS LABEL
LEFT OUTER JOIN (
    SELECT I.Serial_Number, I.Date_Time, I.Test_Result
    FROM
        JP2 AS I
        LEFT OUTER JOIN JP2 AS O
        ON
        (I.Serial_Number=O.Serial_Number) AND
        ((O.Test_Result LIKE '%pass%') AND (O.Test_Result NOT LIKE '%fail%')) AND -- //Gets First Pass!
        (O.Test_Result NOT LIKE '%NO TEST%') AND
        (O.Date_Time<I.Date_Time) -- //(selects FIRST pass or fail)
        -- (I.Date_Time<O.Date_Time) -- //(selects LAST pass or fail)
    WHERE
        (I.Test_Result LIKE '%pass%' AND I.Test_Result NOT LIKE '%fail%') AND -- //Gets First Pass!
        (I.Test_Result NOT LIKE '%NO TEST%') AND
        (O.Serial_Number IS NULL)
) AS DECAY ON 
(LABEL.Serial_Number=DECAY.Serial_Number) AND 
(LABEL.Date_Time<DECAY.Date_Time)
WHERE 
(LABEL.Serial_Number IN (
    SELECT DISTINCT Serial_Number 
    FROM JP1
    WHERE (Serial_Number IS NOT NULL) AND 
    (DATEADD(yy, - 1, GETDATE()) < Date_Time))
)

Я надеюсь, что другие найдут это полезным!

~Джо

0 голосов
/ 17 июля 2010

Правильно ли я считаю, что вам нужна последняя запись test_result для каждой части?

Проверьте ROW_NUMBER () OVER (PARTITION BY ... ORDER BY ...) inдокументы MSDN - это сделает их ослепительно простыми - при условии, что вы используете MSSQL 2005 или выше.

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