oob
oob

Reputation: 1958

Not sure how to go about joining these two tables

I had trouble coming up with a title for this question because I'm not really sure how to do what I want, so I'm not sure if "joining" is the best terminology in this instance.

I have two tables in my database, Plans and Assumptions that look like the following:

Plans Table

Plankey  ass1 ass2 ass3 ass4
aplan    0    6    5    7
bplan    2    0    7    4

Assumptions Table

assType refKey assName
"ass1"  0      "gender factors a"
"ass1"  2      "gender factors b"
"ass2"  0      "age factors a"
"ass2"  6      "age factors b"
"ass3"  5      "inflation factors a"
"ass3"  7      "inflation factors b"
"ass4"  4      "tax factors a"
"ass4"  7      "tax factors b"

I need to make a query (or set of queries and sub-queries) that gives me the names of the assumptions in use for each plan:

Plankey  assName1           assName2         assName3               assName4
aplan    "gender factors a" "age factors b"  "inflation factors a"  "tax factors b"
bplan    "gender factors b" "age factors a"  "inflation factors b"  "tax factors a"

Yeah... I know. assName. Also, even if this isn't the best design, that is beyond my control. I'm just trying to query an existing set of data.

I should also mention there are over 500 assumption types (ass1,ass2,...,ass500,etc) and each type could have over 100 assumption refKey/Names per assumption type.

I'm trying to wrap my head around this and it seems easy, but I just can't figure out how to do it. Any ideas? Maybe there is a concept I'm missing because I just haven't encountered it yet. I'm okay with hardcoding the column names assName1, assName2 etc into my query, but even then I'm unsure how to "lookup" the assNames from the Assumptions table when it seems like I'm looking up from the same table for multiple columns in my result.

EDIT: I ommitted something really important. refkey is re-used in the Assumptions Table. So an assName is uniquely determined by the combination of assType and refKey. I apologize for not making that clear in my example! I forgot about that until I looked at the answers. I have changed my example to reflect this as well.

EDIT2: I am using MS SQL Server.

EDIT3: I expect to find match in the assumptions table for every plan. If not, I would have bigger problems - unrelated to this question though.

Upvotes: 2

Views: 191

Answers (5)

Alex Humphrey
Alex Humphrey

Reputation: 6209

The best way to deal with this would be to turn your assName* columns into rows in a sub-query or a view on that second table.

If you're using SQL Server, you can use UNPIVOT to do it. I'm not sure what other DBMSs support UNPIVOT though, so let us know which one you're using. If you are using SQL Server I'll provide an example.

http://msdn.microsoft.com/en-us/library/ms177410.aspx

UPDATE:

Here's a query that will transform the Plans table into a form that is easy to join to. However, I have overlooked the fact that the result you want is still 'pivoted', so it's not going to be much use unless you can accept the query results in a more 'normalized' form. For the 'pivoted' result, other suggestions are much better.

Here's the query anyway, for anyone that's interested:

SELECT [PVT].[PlanKey], [PVT].[RefKey], [PVT].[Ass]
FROM   [Plans] [P]
       UNPIVOT
       ([RefKey] FOR [Ass] IN ([ass1], [ass2], [ass3], [ass4])) [PVT]

Apologies if it's not of any use - my mistake.

Upvotes: 1

Tom H
Tom H

Reputation: 47444

You could try normalizing the Plans table as an inline query and then joining to that. I don't know which would be worse - 500 unions or 500 joins.

SELECT
    SQ.Plankey,
    A.assName
FROM
(

    SELECT Plankey, ass1 AS assRefKey, 'ass1' AS assType FROM Plans WHERE ass1 IS NOT NULL UNION ALL
    SELECT Plankey, ass2, 'ass2' AS assType FROM Plans WHERE ass2 IS NOT NULL UNION ALL
    ...
    SELECT Plankey, ass500, 'ass500' AS assType FROM Plans WHERE ass500 IS NOT NULL
) SQ
INNER JOIN Assumptions A ON
    A.refKey = SQ.assRefKey AND
    A.assType = SQ.assType

The resultset won't be in the format that you requested, but it might be better for some uses of the data. Either way, that's one ugly database design. You'd think that by the time they got to ass100 that they would have realized that they might be doing something wrong.

If you're going to use this approach, you might want to just create a view for that inner query and query against the view.

Upvotes: 1

OMG Ponies
OMG Ponies

Reputation: 332521

You have to LEFT JOIN onto the ASSUMPTIONS table for every ass# column in the PLANS table:

   SELECT p.plankey,
          a1.assname,
          a2.assname,
          a3.assname,
          a4.assname
     FROM PLANS p
LEFT JOIN ASSUMPTIONS a1 ON a1.refkey = p.ass1
                        AND a1.asstype = 'ass1'
LEFT JOIN ASSUMPTIONS a2 ON a2.refkey = p.ass2
                        AND a2.asstype = 'ass2'
LEFT JOIN ASSUMPTIONS a3 ON a3.refkey = p.ass3
                        AND a3.asstype = 'ass3'
LEFT JOIN ASSUMPTIONS a4 ON a4.refkey = p.ass4
                        AND a4.asstype = 'ass4'

Without knowing the database, I can't provide the syntax for dynamic SQL to construct the query for a varying number of joins that need to be performed.

Upvotes: 6

Gilbert Le Blanc
Gilbert Le Blanc

Reputation: 51445

This would be pretty straightforward in a programming language. You would read the entire Assumptions table, then read the Plans table, substituting the assName for the four refKeys in each Plans row.

OMG Ponies has the most popular SQL answer at the time I'm editing this answer.

Upvotes: 1

Mchl
Mchl

Reputation: 62359

SELECT p.PlanKey, a1.name, a2.name, a3.name, a4.name
FROM Plans AS p
LEFT JOIN Assumptions AS a1 ON p.ass1 = a1.refKey
LEFT JOIN Assumptions AS a2 ON p.ass2 = a2.refKey
LEFT JOIN Assumptions AS a3 ON p.ass3 = a3.refKey
LEFT JOIN Assumptions AS a4 ON p.ass4 = a4.refKey

Upvotes: 1

Related Questions