Get records which satisfy particular condition and exclude others

  • I have the following data with client account decision and status

    Clientfields value

    111Decisionaccept

    111Statusdeposited

    112decisionaccept

    113Decisionaccept

    114Decisionaccept

    114Statuscontract

    115DecisionConditional Accept

    115Statusdeposited

    116DecisionConditional Accept

    117DecisionConditional Accept

    118Statuscontract

    118DecisionConditional Accept

    I would like to get

    records whose decision is 'accept' and also with status ('deposited', 'contract')

    and get the records whose decision is 'conditional accept' and also with status ('deposited', 'contract')

    in two different columns.

    please help in writing case statements for this.

  • sindhupavani123 (11/4/2016)


    I have the following data with client account decision and status

    Clientfields value

    111Decisionaccept

    111Statusdeposited

    112decisionaccept

    113Decisionaccept

    114Decisionaccept

    114Statuscontract

    115DecisionConditional Accept

    115Statusdeposited

    116DecisionConditional Accept

    117DecisionConditional Accept

    118Statuscontract

    118DecisionConditional Accept

    I would like to get

    records whose decision is 'accept' and also with status ('deposited', 'contract')

    and get the records whose decision is 'conditional accept' and also with status ('deposited', 'contract')

    in two different columns.

    please help in writing case statements for this.

    Can you show how you would like the output to look, based on your sample data?

    The absence of evidence is not evidence of absence.
    Martin Rees

    You can lead a horse to water, but a pencil must be lead.
    Stan Laurel

  • Once you normalise your EAV-structured data, it's easy. The best way is to change your database design so that your tables are normalised. If that's not an option (sometimes we just have to work with what we have) then you can use a CTE to normalise the data on the fly, and query the CTE. See this article for how to normalise an EAV design.

    John

  • sindhupavani123 (11/4/2016)


    I have the following data with client account decision and status

    Clientfields value

    111Decisionaccept

    111Statusdeposited

    112decisionaccept

    113Decisionaccept

    114Decisionaccept

    114Statuscontract

    115DecisionConditional Accept

    115Statusdeposited

    116DecisionConditional Accept

    117DecisionConditional Accept

    118Statuscontract

    118DecisionConditional Accept

    I would like to get

    records whose decision is 'accept' and also with status ('deposited', 'contract')

    and get the records whose decision is 'conditional accept' and also with status ('deposited', 'contract')

    in two different columns.

    please help in writing case statements for this.

    Use a standard comparison in the WHERE clause to capture rows matching the required "decision", and an EXISTS check for the status.

    β€œWrite the query the simplest way. If through testing it becomes clear that the performance is inadequate, consider alternative query forms.” - Gail Shaw

    For fast, accurate and documented assistance in answering your questions, please read this article.
    Understanding and using APPLY, (I) and (II) Paul White
    Hidden RBAR: Triangular Joins / The "Numbers" or "Tally" Table: What it is and how it replaces a loop Jeff Moden

  • My output should look like this:

    Clientfields value NewCol

    111Decisionaccept

    111Statusdeposited 111

    112decisionaccept

    113Decisionaccept

    114Decisionaccept

    114Statuscontract 114

    115DecisionConditional Accept

    115Statusdeposited 115

    116DecisionConditional Accept

    117DecisionConditional Accept

    118Statuscontract 118

    118DecisionConditional Accept

    Tried writing the query but I'm unable to get the correct output

  • sindhupavani123 (11/4/2016)


    My output should look like this:

    Clientfields value NewCol

    111Decisionaccept

    111Statusdeposited 111

    112decisionaccept

    113Decisionaccept

    114Decisionaccept

    114Statuscontract 114

    115DecisionConditional Accept

    115Statusdeposited 115

    116DecisionConditional Accept

    117DecisionConditional Accept

    118Statuscontract 118

    118DecisionConditional Accept

    Tried writing the query but I'm unable to get the correct output

    Quick suggestion

    😎

    USE TEEST;

    GO

    SET NOCOUNT ON;

    -- SAMPLE DATA SET

    ;WITH SAMPLE_DATA(Client,fields,value) AS

    ( SELECT X.Client,X.fields,X.value

    FROM ( VALUES

    (111,'Decision','accept')

    ,(111,'Status','deposited')

    ,(112,'decision','accept')

    ,(113,'Decision','accept')

    ,(114,'Decision','accept')

    ,(114,'Status','contract')

    ,(115,'Decision','Conditional Accept')

    ,(115,'Status','deposited')

    ,(116,'Decision','Conditional Accept')

    ,(117,'Decision','Conditional Accept')

    ,(118,'Status','contract')

    ,(118,'Decision','Conditional Accept')

    ) X(Client,fields,value)

    )

    -- CROSS-TAB QUERY

    --;WITH

    ,BASE_DATA AS

    (

    SELECT

    SD.Client

    ,MAX(CASE WHEN SD.fields = 'Status' THEN SD.value END ) AS Status

    ,MAX(CASE WHEN SD.fields = 'Decision' THEN SD.value END ) AS Decision

    FROM SAMPLE_DATA SD

    GROUP BY SD.Client

    )

    ,CONDITIONAL_CLIENTS AS

    (

    SELECT

    BD.Client

    ,BD.Status

    FROM BASE_DATA BD

    WHERE

    (

    BD.Decision = 'accept'

    AND

    BD.Status = 'contract'

    )

    OR

    (

    BD.Decision = 'conditional accept'

    AND

    BD.Status IN ('deposited', 'contract')

    )

    )

    SELECT

    SD.Client

    ,SD.fields

    ,SD.value

    ,CC.Client

    FROM SAMPLE_DATA SD

    LEFT OUTER JOIN CONDITIONAL_CLIENTS CC

    ON SD.Client = CC.Client

    AND SD.value = CC.Status;

    Output

    Client fields value Client

    ----------- -------- ------------------ -------

    111 Decision accept NULL

    111 Status deposited NULL

    112 decision accept NULL

    113 Decision accept NULL

    114 Decision accept NULL

    114 Status contract 114

    115 Decision Conditional Accept NULL

    115 Status deposited 115

    116 Decision Conditional Accept NULL

    117 Decision Conditional Accept NULL

    118 Status contract 118

    118 Decision Conditional Accept NULL

  • ChrisM@Work (11/4/2016)


    sindhupavani123 (11/4/2016)


    I have the following data with client account decision and status

    Clientfields value

    111Decisionaccept

    111Statusdeposited

    112decisionaccept

    113Decisionaccept

    114Decisionaccept

    114Statuscontract

    115DecisionConditional Accept

    115Statusdeposited

    116DecisionConditional Accept

    117DecisionConditional Accept

    118Statuscontract

    118DecisionConditional Accept

    I would like to get

    records whose decision is 'accept' and also with status ('deposited', 'contract')

    and get the records whose decision is 'conditional accept' and also with status ('deposited', 'contract')

    in two different columns.

    please help in writing case statements for this.

    Use a standard comparison in the WHERE clause to capture rows matching the required "decision", and an EXISTS check for the status.

    This output disagrees with your initial post, which stated that you only want the records that have a certain decision and also have a certain status. This post says you want ALL the records, but only create a new field with the Client number in it when those conditions are fulfilled, and only on the status records. Please clarify.

    Steve (aka sgmunson) πŸ™‚ πŸ™‚ πŸ™‚
    Rent Servers for Income (picks and shovels strategy)

  • This also matches the somewhat ambiguous and vague requirements:

    ;WITH SAMPLE_DATA(Client,fields,value) AS

    ( SELECT X.Client,X.fields,X.value

    FROM ( VALUES

    (111,'Decision','accept')

    ,(111,'Status','deposited')

    ,(112,'decision','accept')

    ,(113,'Decision','accept')

    ,(114,'Decision','accept')

    ,(114,'Status','contract')

    ,(115,'Decision','Conditional Accept')

    ,(115,'Status','deposited')

    ,(116,'Decision','Conditional Accept')

    ,(117,'Decision','Conditional Accept')

    ,(118,'Status','contract')

    ,(118,'Decision','Conditional Accept')

    ) X(Client,fields,value)

    )

    SELECT

    b.Client, b.fields, b.[value],

    Result = x.Client

    FROM SAMPLE_DATA b

    OUTER APPLY (

    SELECT i.Client

    FROM SAMPLE_DATA i

    WHERE i.Client = b.Client

    AND i.fields = 'Decision'

    AND i.[Value] IN ('accept', 'Conditional Accept')

    AND b.fields = 'Status'

    AND b.[Value] IN ('deposited','contract')

    ) x

    Thanks EE for the sample data setup πŸ˜‰

    β€œWrite the query the simplest way. If through testing it becomes clear that the performance is inadequate, consider alternative query forms.” - Gail Shaw

    For fast, accurate and documented assistance in answering your questions, please read this article.
    Understanding and using APPLY, (I) and (II) Paul White
    Hidden RBAR: Triangular Joins / The "Numbers" or "Tally" Table: What it is and how it replaces a loop Jeff Moden

  • Eirikur Eiriksson (11/4/2016)


    sindhupavani123 (11/4/2016)


    My output should look like this:

    Clientfields value NewCol

    111Decisionaccept

    111Statusdeposited 111

    112decisionaccept

    113Decisionaccept

    114Decisionaccept

    114Statuscontract 114

    115DecisionConditional Accept

    115Statusdeposited 115

    116DecisionConditional Accept

    117DecisionConditional Accept

    118Statuscontract 118

    118DecisionConditional Accept

    Tried writing the query but I'm unable to get the correct output

    Quick suggestion

    😎

    USE TEEST;

    GO

    SET NOCOUNT ON;

    -- SAMPLE DATA SET

    ;WITH SAMPLE_DATA(Client,fields,value) AS

    ( SELECT X.Client,X.fields,X.value

    FROM ( VALUES

    (111,'Decision','accept')

    ,(111,'Status','deposited')

    ,(112,'decision','accept')

    ,(113,'Decision','accept')

    ,(114,'Decision','accept')

    ,(114,'Status','contract')

    ,(115,'Decision','Conditional Accept')

    ,(115,'Status','deposited')

    ,(116,'Decision','Conditional Accept')

    ,(117,'Decision','Conditional Accept')

    ,(118,'Status','contract')

    ,(118,'Decision','Conditional Accept')

    ) X(Client,fields,value)

    )

    -- CROSS-TAB QUERY

    --;WITH

    ,BASE_DATA AS

    (

    SELECT

    SD.Client

    ,MAX(CASE WHEN SD.fields = 'Status' THEN SD.value END ) AS Status

    ,MAX(CASE WHEN SD.fields = 'Decision' THEN SD.value END ) AS Decision

    FROM SAMPLE_DATA SD

    GROUP BY SD.Client

    )

    ,CONDITIONAL_CLIENTS AS

    (

    SELECT

    BD.Client

    ,BD.Status

    FROM BASE_DATA BD

    WHERE

    (

    BD.Decision = 'accept'

    AND

    BD.Status = 'contract'

    )

    OR

    (

    BD.Decision = 'conditional accept'

    AND

    BD.Status IN ('deposited', 'contract')

    )

    )

    SELECT

    SD.Client

    ,SD.fields

    ,SD.value

    ,CC.Client

    FROM SAMPLE_DATA SD

    LEFT OUTER JOIN CONDITIONAL_CLIENTS CC

    ON SD.Client = CC.Client

    AND SD.value = CC.Status;

    Output

    Client fields value Client

    ----------- -------- ------------------ -------

    111 Decision accept NULL

    111 Status deposited NULL

    112 decision accept NULL

    113 Decision accept NULL

    114 Decision accept NULL

    114 Status contract 114

    115 Decision Conditional Accept NULL

    115 Status deposited 115

    116 Decision Conditional Accept NULL

    117 Decision Conditional Accept NULL

    118 Status contract 118

    118 Decision Conditional Accept NULL

    Your output misses Client 111, who does meet the required conditions. However, the original poster also has stated output that disagrees with their initial statement, and I've asked for clarification.

    Steve (aka sgmunson) πŸ™‚ πŸ™‚ πŸ™‚
    Rent Servers for Income (picks and shovels strategy)

  • sgmunson (11/4/2016)


    Your output misses Client 111, who does meet the required conditions. However, the original poster also has stated output that disagrees with their initial statement, and I've asked for clarification.

    Missed the 'deposited' in the where clause, simply change

    BD.Status = 'contract'

    to

    BD.Status IN ('deposited', 'contract')

    😎

    Corrected query

    USE TEEST;

    GO

    SET NOCOUNT ON;

    -- SAMPLE DATA SET

    ;WITH SAMPLE_DATA(Client,fields,value) AS

    ( SELECT X.Client,X.fields,X.value

    FROM ( VALUES

    (111,'Decision','accept')

    ,(111,'Status','deposited')

    ,(112,'decision','accept')

    ,(113,'Decision','accept')

    ,(114,'Decision','accept')

    ,(114,'Status','contract')

    ,(115,'Decision','Conditional Accept')

    ,(115,'Status','deposited')

    ,(116,'Decision','Conditional Accept')

    ,(117,'Decision','Conditional Accept')

    ,(118,'Status','contract')

    ,(118,'Decision','Conditional Accept')

    ) X(Client,fields,value)

    )

    -- CROSS-TAB QUERY

    --;WITH

    ,BASE_DATA AS

    (

    SELECT

    SD.Client

    ,MAX(CASE WHEN SD.fields = 'Status' THEN SD.value END ) AS Status

    ,MAX(CASE WHEN SD.fields = 'Decision' THEN SD.value END ) AS Decision

    FROM SAMPLE_DATA SD

    GROUP BY SD.Client

    )

    ,CONDITIONAL_CLIENTS AS

    (

    SELECT

    BD.Client

    ,BD.Status

    FROM BASE_DATA BD

    WHERE

    (

    BD.Decision = 'accept'

    AND

    BD.Status IN ('deposited', 'contract')

    )

    OR

    (

    BD.Decision = 'conditional accept'

    AND

    BD.Status IN ('deposited', 'contract')

    )

    )

    SELECT

    SD.Client

    ,SD.fields

    ,SD.value

    ,CC.Client

    FROM SAMPLE_DATA SD

    LEFT OUTER JOIN CONDITIONAL_CLIENTS CC

    ON SD.Client = CC.Client

    AND SD.value = CC.Status;

    Output

    Client fields value Client

    ----------- -------- ------------------ ---------

    111 Decision accept NULL

    111 Status deposited 111

    112 decision accept NULL

    113 Decision accept NULL

    114 Decision accept NULL

    114 Status contract 114

    115 Decision Conditional Accept NULL

    115 Status deposited 115

    116 Decision Conditional Accept NULL

    117 Decision Conditional Accept NULL

    118 Status contract 118

    118 Decision Conditional Accept NULL

  • And here's an alternative that achieves the desired output (however conflicting with the original post that may be...), and shows the performance characteristics, as I'm guessing this sample data would likely be much more extensive in the real world, and may well have other values not shown here...

    CREATE TABLE #TEST_DATA (

    Client int,

    Fields varchar(30),

    Value varchar(30)

    );

    CREATE NONCLUSTERED INDEX IX_TESTDATA_Client_INCLUDE_Fields_Value ON #TEST_DATA

    (

    Client ASC

    )

    INCLUDE (Fields, Value);

    CREATE NONCLUSTERED INDEX IX_TESTDATA_Fields_Value_INCLUDE_Client ON #TEST_DATA

    (

    Fields ASC,

    Value ASC

    )

    INCLUDE (Client);

    INSERT INTO #TEST_DATA (Client, Fields, Value)

    VALUES(111, 'Decision', 'accept'),

    (111, 'Status', 'deposited'),

    (112, 'decision', 'accept'),

    (113, 'Decision', 'accept'),

    (114, 'Decision', 'accept'),

    (114, 'Status', 'contract'),

    (115, 'Decision', 'Conditional Accept'),

    (115, 'Status', 'deposited'),

    (116, 'Decision', 'Conditional Accept'),

    (117, 'Decision', 'Conditional Accept'),

    (118, 'Status', 'contract'),

    (118, 'Decision', 'Conditional Accept');

    DBCC FREEPROCCACHE;

    SET STATISTICS IO, TIME ON;

    WITH ALL_DATA AS (

    SELECT *,

    ROW_NUMBER() OVER(PARTITION BY Client ORDER BY CASE Fields WHEN 'Decision' THEN 1 WHEN 'Status' THEN 2 ELSE 9999 END, Value) AS RN

    FROM #TEST_DATA

    )

    SELECT A1.Client, A1.Fields, A1.Value,

    CASE

    WHEN (

    (A1.Fields = 'Decision' AND A1.Value IN ('accept', 'Conditional Accept'))

    AND

    (A2.Fields = 'Status' AND A2.Value IN ('contract', 'deposited'))

    )

    OR

    (

    (A2.Fields = 'Decision' AND A2.Value IN ('accept', 'Conditional Accept'))

    AND

    (A1.Fields = 'Status' AND A1.Value IN ('contract', 'deposited'))

    ) THEN A1.Client

    ELSE NULL

    END AS NewCol

    FROM ALL_DATA AS A1

    LEFT OUTER JOIN ALL_DATA AS A2

    ON A1.Client = A2.Client

    AND A1.RN = 2

    AND A2.RN = 1

    ORDER BY A1.Client, A1.Fields;

    SET STATISTICS TIME, IO OFF;

    DROP TABLE #TEST_DATA;

    Steve (aka sgmunson) πŸ™‚ πŸ™‚ πŸ™‚
    Rent Servers for Income (picks and shovels strategy)

  • ChrisM@Work (11/4/2016)


    This also matches the somewhat ambiguous and vague requirements:

    ;WITH SAMPLE_DATA(Client,fields,value) AS

    ( SELECT X.Client,X.fields,X.value

    FROM ( VALUES

    (111,'Decision','accept')

    ,(111,'Status','deposited')

    ,(112,'decision','accept')

    ,(113,'Decision','accept')

    ,(114,'Decision','accept')

    ,(114,'Status','contract')

    ,(115,'Decision','Conditional Accept')

    ,(115,'Status','deposited')

    ,(116,'Decision','Conditional Accept')

    ,(117,'Decision','Conditional Accept')

    ,(118,'Status','contract')

    ,(118,'Decision','Conditional Accept')

    ) X(Client,fields,value)

    )

    SELECT

    b.Client, b.fields, b.[value],

    Result = x.Client

    FROM SAMPLE_DATA b

    OUTER APPLY (

    SELECT i.Client

    FROM SAMPLE_DATA i

    WHERE i.Client = b.Client

    AND i.fields = 'Decision'

    AND i.[Value] IN ('accept', 'Conditional Accept')

    AND b.fields = 'Status'

    AND b.[Value] IN ('deposited','contract')

    ) x

    Interestingly the Filter operator (where clause) cost in the execution plan jumps up from 7% to 64% when combining all the conditions instead of targeting and combining narrower conditions.

    😎

    Didn't test this case specifically though and the costs in the execution plans and comparison between those can be a little misleading.

    Thanks EE for the sample data setup πŸ˜‰

    You are very welcome Chris πŸ˜€

  • sgmunson (11/4/2016)


    And here's an alternative that achieves the desired output (however conflicting with the original post that may be...), and shows the performance characteristics, as I'm guessing this sample data would likely be much more extensive in the real world, and may well have other values not shown here...

    CREATE TABLE #TEST_DATA (

    Client int,

    Fields varchar(30),

    Value varchar(30)

    );

    CREATE NONCLUSTERED INDEX IX_TESTDATA_Client_INCLUDE_Fields_Value ON #TEST_DATA

    (

    Client ASC

    )

    INCLUDE (Fields, Value);

    CREATE NONCLUSTERED INDEX IX_TESTDATA_Fields_Value_INCLUDE_Client ON #TEST_DATA

    (

    Fields ASC,

    Value ASC

    )

    INCLUDE (Client);

    INSERT INTO #TEST_DATA (Client, Fields, Value)

    VALUES(111, 'Decision', 'accept'),

    (111, 'Status', 'deposited'),

    (112, 'decision', 'accept'),

    (113, 'Decision', 'accept'),

    (114, 'Decision', 'accept'),

    (114, 'Status', 'contract'),

    (115, 'Decision', 'Conditional Accept'),

    (115, 'Status', 'deposited'),

    (116, 'Decision', 'Conditional Accept'),

    (117, 'Decision', 'Conditional Accept'),

    (118, 'Status', 'contract'),

    (118, 'Decision', 'Conditional Accept');

    DBCC FREEPROCCACHE;

    SET STATISTICS IO, TIME ON;

    WITH ALL_DATA AS (

    SELECT *,

    ROW_NUMBER() OVER(PARTITION BY Client ORDER BY CASE Fields WHEN 'Decision' THEN 1 WHEN 'Status' THEN 2 ELSE 9999 END, Value) AS RN

    FROM #TEST_DATA

    )

    SELECT A1.Client, A1.Fields, A1.Value,

    CASE

    WHEN (

    (A1.Fields = 'Decision' AND A1.Value IN ('accept', 'Conditional Accept'))

    AND

    (A2.Fields = 'Status' AND A2.Value IN ('contract', 'deposited'))

    )

    OR

    (

    (A2.Fields = 'Decision' AND A2.Value IN ('accept', 'Conditional Accept'))

    AND

    (A1.Fields = 'Status' AND A1.Value IN ('contract', 'deposited'))

    ) THEN A1.Client

    ELSE NULL

    END AS NewCol

    FROM ALL_DATA AS A1

    LEFT OUTER JOIN ALL_DATA AS A2

    ON A1.Client = A2.Client

    AND A1.RN = 2

    AND A2.RN = 1

    ORDER BY A1.Client, A1.Fields;

    SET STATISTICS TIME, IO OFF;

    DROP TABLE #TEST_DATA;

    Nice in the terms of only doing 2 scans but the down side is that the execution plan has two sort operators, on for each scan which are quite costly in the absence of a POC index for the ROW_NUMBER function.

    😎

  • Thank you all for the help.

    What should i do if i want two columns with below output?

    Client fields value ClientClient

    ----------- -------- ------------------ ---------

    111 Decision accept NULL

    111 Status deposited 111

    112 decision accept NULL

    113 Decision accept NULL

    114 Decision accept NULL

    114 Status contract 114

    115 Decision Conditional Accept Null

    115 Status deposited 115

    116 Decision Conditional Accept NULL

    117 Decision Conditional Accept NULL

    118 Status contract 118

    118 Decision Conditional Accept NULL

  • Thank you for help.

    What should i do to get two columns with below expected output.

    Client fields value Client1Client2

    ----------- -------- ------------------ ---------

    111 Decision accept NULL

    111 Status deposited 111

    112 decision accept NULL

    113 Decision accept NULL

    114 Decision accept NULL

    114 Status contract 114

    115 Decision Conditional Accept Null

    115 Status deposited 115

    116 Decision Conditional Accept NULL

    117 Decision Conditional Accept NULL

    118 Status contract 118

    118 Decision Conditional Accept NULL

Viewing 15 posts - 1 through 15 (of 19 total)

You must be logged in to reply to this topic. Login to reply