Help avoiding Dynamic SQL if possible?

  • I'm currently writing some code for a task at work where I need to return values for individual ID's stating what there value is when being passed some parameters. The problem I have is that parameter list was currently set at 4 and now have asked to increase from 4 to 8 with the possibility of further expansion at a later date.

    Previously I had a lookup table with all the 4 values and then used OUTER APPLY for these 4 to give the results I needed. I can expand the 4 to 8 to get the results again but if it jumps to 30 params then I don't want 30 OUTER APPLY's.

    I know we can do this dynamically (which I've never written dynamic code this way before so i'll need to learn) but is there any other way this can be achieved or is Dynamic the only way to go.

    CREATE TABLE #TestMapping

    (

    ID INT IDENTITY(1,1),

    ValueID INT,

    IsEnabled BIT,

    CreatedDateTime DATETIME

    )

    INSERT INTO #TestMapping (ValueID, IsEnabled, CreatedDateTime)

    VALUES

    (10, 1, GETDATE()),

    (20, 1, GETDATE()),

    (30, 1, GETDATE()),

    (40, 1, GETDATE()),

    SELECT TOP 10

    table1.ID,

    oa1.Result

    FROM

    dbo.table1

    OUTER APPLY (

    SELECT TOP 1 Result

    FROM dbo.table2

    INNER JOIN #TestMapping tm

    ON table2.ValueID= tm.ValueID

    WHERE

    table2.ValueID= 1

    ) oa1

    Sample code above of how I'm currently doing it.

    Any help would be appreciated.

  • Have a look here[/url].

    Dynamic SQL is a good option in this case.

    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

  • Thanks for the quick reply. I thought it was the case but thought I'll see if anything else could be done.

    I'll have a look at that link and revert back if any questions. I'm sure I will.

  • Phil,

    I've had a quick read, I'm a little confused... The values of 4,8,30,N... will be passed via the #temp table i created.

    Since I'm using an outer apply in this case and I'm selecting a top 1 - I need to pass the ID from the temp table to give me the result i want. If i don't i return the same result.

    How can i call that query dynamically by passing through the ID needed.

  • I've updated the code to show what i mean with multiple outers. Sorry for the confusion.

    CREATE TABLE #TestMapping

    (

    ID INT IDENTITY(1,1),

    ValueID INT,

    IsEnabled BIT,

    CreatedDateTime DATETIME

    )

    INSERT INTO #TestMapping (ValueID, IsEnabled, CreatedDateTime)

    VALUES

    (10, 1, GETDATE()),

    (20, 1, GETDATE()),

    (30, 1, GETDATE()),

    (40, 1, GETDATE()),

    DECLARE @ValueID1 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 1 AND IsEnabled = 1)

    DECLARE @ValueID2 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 2 AND IsEnabled = 1)

    DECLARE @ValueID3 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 3 AND IsEnabled = 1)

    DECLARE @ValueID4 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 4 AND IsEnabled = 1)

    SELECT TOP 10

    table1.ID,

    oa1.Result

    oa2.Result

    oa3.Result

    FROM

    dbo.table1

    OUTER APPLY (

    SELECT TOP 1 Result

    FROM dbo.table2

    INNER JOIN #TestMapping tm

    ON table2.ValueID= tm.ValueID

    WHERE

    table2.ValueID= @ValueID1

    ) oa1

    OUTER APPLY (

    SELECT TOP 1 Result

    FROM dbo.table2

    INNER JOIN #TestMapping tm

    ON table2.ValueID= tm.ValueID

    WHERE

    table2.ValueID= @ValueID2

    ) oa2

    OUTER APPLY (

    SELECT TOP 1 Result

    FROM dbo.table2

    INNER JOIN #TestMapping tm

    ON table2.ValueID= tm.ValueID

    WHERE

    table2.ValueID= @ValueID3

    ) oa3

  • Tava (4/4/2016)


    Phil,

    I've had a quick read, I'm a little confused... The values of 4,8,30,N... will be passed via the #temp table i created.

    Since I'm using an outer apply in this case and I'm selecting a top 1 - I need to pass the ID from the temp table to give me the result i want. If i don't i return the same result.

    How can i call that query dynamically by passing through the ID needed.

    You were right to be confused. I answered the question I thought I was reading ... not what you actually wrote!

    Having said that, despite your code samples, I do not understand what is required here. Though I am almost certain that there will be a much better way.

    What does your proc's current list of arguments look like? Is it just a list of numbers? Do you pass it as a table?

    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

  • Phil Parkin (4/4/2016)


    Tava (4/4/2016)


    Phil,

    I've had a quick read, I'm a little confused... The values of 4,8,30,N... will be passed via the #temp table i created.

    Since I'm using an outer apply in this case and I'm selecting a top 1 - I need to pass the ID from the temp table to give me the result i want. If i don't i return the same result.

    How can i call that query dynamically by passing through the ID needed.

    You were right to be confused. I answered the question I thought I was reading ... not what you actually wrote!

    Having said that, despite your code samples, I do not understand what is required here. Though I am almost certain that there will be a much better way.

    What does your proc's current list of arguments look like? Is it just a list of numbers? Do you pass it as a table?

    I updated the test sample above, i just wrote it quick and dirty thinking it would be easy to understand. The values come from the Temp (which in fact will be a proper lookup) and I'm passing a local variable to capture that and use it in the outer apply. Hopefully that makes more sense.

    pretty much you will be adding a new local variable each time a entry is created (unless you update the table entry) its more when a 5th entry gets added then the Query itself will need to include a futher outer apply.

  • So am I right in thinking that the number of columns which you are returning depends on the number of rows in the temp table? There's some unusual stuff going on here.

    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

  • Phil Parkin (4/4/2016)


    So am I right in thinking that the number of columns which you are returning depends on the number of rows in the temp table? There's some unusual stuff going on here.

    yeap exactly, so imagine there are ValueID's added in the temp, provided all 20 are enabled i would expect to have a 20 outer apply scenario. Returning who the ID belows to and then 20 columns showing each value per ValueID.

    if it was locked to 2/3/4 values, 4 outer applys would be sufficient but in this case there could be 20. hence the need to try and write it in a way it will generate regardless of how many Rows there are in the temp table.

  • OK, let me try again with this post.

    I have some questions about your sample code. Looking only up to the end of the first APPLY:

    select top 10

    table1.ID

    ,oa1.Result

    from dbo.table1

    outer apply (select top 1

    Result

    from dbo.table2

    inner join #TestMapping tm on table2.ValueID = tm.ValueID

    where table2.ValueID = @ValueID1

    ) oa1;

    Firstly: Select top without an ORDER BY means that a semi-random row will be returned.

    Next, there is no relation between table1 and the other two tables. So the same results will be repeated for every row in table1.

    Finally, 'Result' is not in the temp table, so it must be in table2. But if that's the case, doesn't the inner query become this?

    select Result from table2 t2 where t2.ValueId = @ValueId1

    I left out the 'TOP 1' because (as mentioned above) it makes little sense without an ORDER BY.

    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

  • Firstly to answer your questions.

    1. ORDER BY is in my code, i had it left out...

    2. The relationship between Table1 & Table2 exists i left out a condition in my sample code as i thought it might not be relevant.

    CREATE TABLE #TestMapping

    (

    ID INT IDENTITY(1,1),

    ValueID INT,

    ResolutionTypeID INT,

    IsEnabled BIT,

    CreatedDateTime DATETIME

    )

    INSERT INTO #TestMapping (ValueID, ResolutionTypeID, IsEnabled, CreatedDateTime)

    VALUES

    (10,11, 1, GETDATE()),

    (20,11, 1, GETDATE()),

    (30,12, 1, GETDATE()),

    (40,2, 1, GETDATE()),

    DECLARE @ValueID1 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 1 AND IsEnabled = 1)

    DECLARE @ValueID2 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 2 AND IsEnabled = 1)

    DECLARE @ValueID3 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 3 AND IsEnabled = 1)

    DECLARE @ValueID4 INT = (SELECT ValueID FROM #TestMapping WHERE ID = 4 AND IsEnabled = 1)

    SELECT TOP 10

    table1.ID,

    oa1.Result

    FROM

    dbo.table1

    OUTER APPLY (

    SELECT TOP 1 Result

    FROM dbo.table2

    INNER JOIN #TestMapping tm

    ON table2.ValueID= tm.ValueID

    WHERE

    table2.ValueID= @ValueID2

    AND

    table2.SourceID= (

    CASE

    WHEN tm.ResolutionTypeID = 11 THEN table1.ABC

    WHEN tm.ResolutionTypeID = 2 THEN table1.DEF

    WHEN tm.ResolutionTypeID = 12 THEN table1.GHI

    END

    )

    ORDER BY

    table1.ID

    ) oa1

    does this make more sense?

Viewing 11 posts - 1 through 10 (of 10 total)

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