Separate a column into its components based on another tableQuery to normalize table/combine row textHow to...
What are the consequences for a developed nation to not accept any refugees?
Write a function
As a supervisor, what feedback would you expect from a PhD who quits?
Deck of Cards with Shuffle and Sort functionality
What does "spinning upon the shoals" mean?
Moving millions of files to a different directory with specfic name patterns
Can a landlord force all residents to use the landlord's in-house debit card accounts?
My professor has told me he will be the corresponding author. Will it hurt my future career?
How do I talk to my wife about unrealistic expectations?
This LM317 diagram doesn't make any sense to me
Was it ever illegal to name a pig "Napoleon" in France?
Where are the Wazirs?
When do flights get cancelled due to fog?
Would denouncing cheaters from an exam make me less likely to receive penalties?
Chilling water in copper vessel
Category-theoretic treatment of diffs, patches and merging?
Gory anime with pink haired girl escaping an asylum
Is it ok for parents to kiss and romance with each other while their 2- to 8-year-old child watches?
How do I separate enchants from items?
What are the effects of abstaining from eating a certain flavor?
How do ballistic trajectories work in a ring world?
Writing an ace/aro character?
Why the Cauchy Distribution is so useful?
Blocks from @ jafe
Separate a column into its components based on another table
Query to normalize table/combine row textHow to add/update a column with an incremented value and reset said value based on another column in SQLColumn Name in separate table SQL ServerSQL Server query problem when selecting data from child table based on column in parent tableWhy does a table of varchar(255) columns take up less space than an identical table using the correct data typesMerging table output into a column in another tableAdding an IF based comparison columnHow can I optimize this MYSQL Script For many Records?Implicit Conversion of VARCHAR Column to NVARCHAR does not cause expected table scanCheck and count if a column value is used in another column of same table
.everyoneloves__top-leaderboard:empty,.everyoneloves__mid-leaderboard:empty,.everyoneloves__bot-mid-leaderboard:empty{ margin-bottom:0;
}
I'm struggling with a very difficult and complicated query here and I need your help.
I have two tables as you can see below:
1) ACCOUNT1:(ATYPCODE int, ATYPDESC varchar(50), ATLASTFLAG varchar(50))
some example data of this table are:
ATYPCODE ATYPDESC ATLASTFLAG
3000 A 0
19 B 0
1170 C 0
1178 D 1
4000 AA 0
18 BB 0
2020 CC 1
Column ATLASTFLAG
shows whether this record is the last level .
Table number two is
2) ACCOUNT2:(ATYPCODE int, AGLTCODE varchar(50), AGLTLVL int)
some example data of this table are:
ATYPCODE AGLTCODE AGLTLVL
3000 3000 1
19 3000 19 2
1170 3000 191170 3
1178 3000 1911701178 4
4000 4000 1
18 4000 18 2
2020 4000 182020 3
Column ATYPCODE
is exactly the same as ATYPCODE
column in the first table.Column AGLTCODE
is bult based on the concatenation of ATYPCODE
column as you can see. The point is that the second level added to the first level with two spaces . so we have 3000+space+space+19+Other_levels
and we have to separate this column exactly with 4 characters.
and the column AGLTLVL
shows the level of each ATYPCODE
.
what we want to see as a result is this :
ATYPCODE AGLTCODE AGLTLVL ATYPCODE1 ATYPCODE2 ATYPCODE3 ATYPCODE4 ATYPDESC1 ATYPDESC2 ATYPDESC3 ATYPDESC4
and values for each column
ATYPCODE = 1178
AGLTCODE = 3000 1911701178
AGLTLVL = 4
ATYPCODE1 = 3000
ATYPCODE2 = 19
ATYPCODE3 = 1170
ATYPCODE4 = 1178
ATYPDESC1 = A
ATYPDESC2 = B
ATYPDESC3 = C
ATYPDESC4 = D
and also the next record is
ATYPCODE = 2020
AGLTCODE = 4000 182020
AGLTLVL = 3
ATYPCODE1 = 4000
ATYPCODE2 = 18
ATYPCODE3 = 2020
ATYPCODE4 = 2020(last value is repeated)
ATYPDESC1 = AA
ATYPDESC2 = BB
ATYPDESC3 = CC
ATYPDESC4 = CC(Last Values is repeated)
as you can see some records have 4 levels and some have 3 levels
since the structure of the destination table is fixed,for those with 3 levels , last value ATYPCODE3 and ATYPDESC3
should be repeated for columns ATYPCODE4 and ATYPDESC4
.
the only part of query I was able to write was this cause we only nees to store the last level and nothing comes to mind for the rest of the query
SELECT b.ATYPCODE , b.AGLTCODE , b.AGLTLVL
FROM ACCOUNT1 a inner join
ACCOUNT2 b on a.ATYPCODE = b.ATYPCODE
where a.ATLASTFLAG = 1
sql-server sql-server-2014 query
add a comment |
I'm struggling with a very difficult and complicated query here and I need your help.
I have two tables as you can see below:
1) ACCOUNT1:(ATYPCODE int, ATYPDESC varchar(50), ATLASTFLAG varchar(50))
some example data of this table are:
ATYPCODE ATYPDESC ATLASTFLAG
3000 A 0
19 B 0
1170 C 0
1178 D 1
4000 AA 0
18 BB 0
2020 CC 1
Column ATLASTFLAG
shows whether this record is the last level .
Table number two is
2) ACCOUNT2:(ATYPCODE int, AGLTCODE varchar(50), AGLTLVL int)
some example data of this table are:
ATYPCODE AGLTCODE AGLTLVL
3000 3000 1
19 3000 19 2
1170 3000 191170 3
1178 3000 1911701178 4
4000 4000 1
18 4000 18 2
2020 4000 182020 3
Column ATYPCODE
is exactly the same as ATYPCODE
column in the first table.Column AGLTCODE
is bult based on the concatenation of ATYPCODE
column as you can see. The point is that the second level added to the first level with two spaces . so we have 3000+space+space+19+Other_levels
and we have to separate this column exactly with 4 characters.
and the column AGLTLVL
shows the level of each ATYPCODE
.
what we want to see as a result is this :
ATYPCODE AGLTCODE AGLTLVL ATYPCODE1 ATYPCODE2 ATYPCODE3 ATYPCODE4 ATYPDESC1 ATYPDESC2 ATYPDESC3 ATYPDESC4
and values for each column
ATYPCODE = 1178
AGLTCODE = 3000 1911701178
AGLTLVL = 4
ATYPCODE1 = 3000
ATYPCODE2 = 19
ATYPCODE3 = 1170
ATYPCODE4 = 1178
ATYPDESC1 = A
ATYPDESC2 = B
ATYPDESC3 = C
ATYPDESC4 = D
and also the next record is
ATYPCODE = 2020
AGLTCODE = 4000 182020
AGLTLVL = 3
ATYPCODE1 = 4000
ATYPCODE2 = 18
ATYPCODE3 = 2020
ATYPCODE4 = 2020(last value is repeated)
ATYPDESC1 = AA
ATYPDESC2 = BB
ATYPDESC3 = CC
ATYPDESC4 = CC(Last Values is repeated)
as you can see some records have 4 levels and some have 3 levels
since the structure of the destination table is fixed,for those with 3 levels , last value ATYPCODE3 and ATYPDESC3
should be repeated for columns ATYPCODE4 and ATYPDESC4
.
the only part of query I was able to write was this cause we only nees to store the last level and nothing comes to mind for the rest of the query
SELECT b.ATYPCODE , b.AGLTCODE , b.AGLTLVL
FROM ACCOUNT1 a inner join
ACCOUNT2 b on a.ATYPCODE = b.ATYPCODE
where a.ATLASTFLAG = 1
sql-server sql-server-2014 query
add a comment |
I'm struggling with a very difficult and complicated query here and I need your help.
I have two tables as you can see below:
1) ACCOUNT1:(ATYPCODE int, ATYPDESC varchar(50), ATLASTFLAG varchar(50))
some example data of this table are:
ATYPCODE ATYPDESC ATLASTFLAG
3000 A 0
19 B 0
1170 C 0
1178 D 1
4000 AA 0
18 BB 0
2020 CC 1
Column ATLASTFLAG
shows whether this record is the last level .
Table number two is
2) ACCOUNT2:(ATYPCODE int, AGLTCODE varchar(50), AGLTLVL int)
some example data of this table are:
ATYPCODE AGLTCODE AGLTLVL
3000 3000 1
19 3000 19 2
1170 3000 191170 3
1178 3000 1911701178 4
4000 4000 1
18 4000 18 2
2020 4000 182020 3
Column ATYPCODE
is exactly the same as ATYPCODE
column in the first table.Column AGLTCODE
is bult based on the concatenation of ATYPCODE
column as you can see. The point is that the second level added to the first level with two spaces . so we have 3000+space+space+19+Other_levels
and we have to separate this column exactly with 4 characters.
and the column AGLTLVL
shows the level of each ATYPCODE
.
what we want to see as a result is this :
ATYPCODE AGLTCODE AGLTLVL ATYPCODE1 ATYPCODE2 ATYPCODE3 ATYPCODE4 ATYPDESC1 ATYPDESC2 ATYPDESC3 ATYPDESC4
and values for each column
ATYPCODE = 1178
AGLTCODE = 3000 1911701178
AGLTLVL = 4
ATYPCODE1 = 3000
ATYPCODE2 = 19
ATYPCODE3 = 1170
ATYPCODE4 = 1178
ATYPDESC1 = A
ATYPDESC2 = B
ATYPDESC3 = C
ATYPDESC4 = D
and also the next record is
ATYPCODE = 2020
AGLTCODE = 4000 182020
AGLTLVL = 3
ATYPCODE1 = 4000
ATYPCODE2 = 18
ATYPCODE3 = 2020
ATYPCODE4 = 2020(last value is repeated)
ATYPDESC1 = AA
ATYPDESC2 = BB
ATYPDESC3 = CC
ATYPDESC4 = CC(Last Values is repeated)
as you can see some records have 4 levels and some have 3 levels
since the structure of the destination table is fixed,for those with 3 levels , last value ATYPCODE3 and ATYPDESC3
should be repeated for columns ATYPCODE4 and ATYPDESC4
.
the only part of query I was able to write was this cause we only nees to store the last level and nothing comes to mind for the rest of the query
SELECT b.ATYPCODE , b.AGLTCODE , b.AGLTLVL
FROM ACCOUNT1 a inner join
ACCOUNT2 b on a.ATYPCODE = b.ATYPCODE
where a.ATLASTFLAG = 1
sql-server sql-server-2014 query
I'm struggling with a very difficult and complicated query here and I need your help.
I have two tables as you can see below:
1) ACCOUNT1:(ATYPCODE int, ATYPDESC varchar(50), ATLASTFLAG varchar(50))
some example data of this table are:
ATYPCODE ATYPDESC ATLASTFLAG
3000 A 0
19 B 0
1170 C 0
1178 D 1
4000 AA 0
18 BB 0
2020 CC 1
Column ATLASTFLAG
shows whether this record is the last level .
Table number two is
2) ACCOUNT2:(ATYPCODE int, AGLTCODE varchar(50), AGLTLVL int)
some example data of this table are:
ATYPCODE AGLTCODE AGLTLVL
3000 3000 1
19 3000 19 2
1170 3000 191170 3
1178 3000 1911701178 4
4000 4000 1
18 4000 18 2
2020 4000 182020 3
Column ATYPCODE
is exactly the same as ATYPCODE
column in the first table.Column AGLTCODE
is bult based on the concatenation of ATYPCODE
column as you can see. The point is that the second level added to the first level with two spaces . so we have 3000+space+space+19+Other_levels
and we have to separate this column exactly with 4 characters.
and the column AGLTLVL
shows the level of each ATYPCODE
.
what we want to see as a result is this :
ATYPCODE AGLTCODE AGLTLVL ATYPCODE1 ATYPCODE2 ATYPCODE3 ATYPCODE4 ATYPDESC1 ATYPDESC2 ATYPDESC3 ATYPDESC4
and values for each column
ATYPCODE = 1178
AGLTCODE = 3000 1911701178
AGLTLVL = 4
ATYPCODE1 = 3000
ATYPCODE2 = 19
ATYPCODE3 = 1170
ATYPCODE4 = 1178
ATYPDESC1 = A
ATYPDESC2 = B
ATYPDESC3 = C
ATYPDESC4 = D
and also the next record is
ATYPCODE = 2020
AGLTCODE = 4000 182020
AGLTLVL = 3
ATYPCODE1 = 4000
ATYPCODE2 = 18
ATYPCODE3 = 2020
ATYPCODE4 = 2020(last value is repeated)
ATYPDESC1 = AA
ATYPDESC2 = BB
ATYPDESC3 = CC
ATYPDESC4 = CC(Last Values is repeated)
as you can see some records have 4 levels and some have 3 levels
since the structure of the destination table is fixed,for those with 3 levels , last value ATYPCODE3 and ATYPDESC3
should be repeated for columns ATYPCODE4 and ATYPDESC4
.
the only part of query I was able to write was this cause we only nees to store the last level and nothing comes to mind for the rest of the query
SELECT b.ATYPCODE , b.AGLTCODE , b.AGLTLVL
FROM ACCOUNT1 a inner join
ACCOUNT2 b on a.ATYPCODE = b.ATYPCODE
where a.ATLASTFLAG = 1
sql-server sql-server-2014 query
sql-server sql-server-2014 query
edited 9 hours ago
Paul White♦
57.4k15 gold badges302 silver badges475 bronze badges
57.4k15 gold badges302 silver badges475 bronze badges
asked 11 hours ago
Pantea TourangPantea Tourang
47914 bronze badges
47914 bronze badges
add a comment |
add a comment |
1 Answer
1
active
oldest
votes
The first thing I would do is create some keys on the source tables:
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT1 (ATYPCODE);
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT2 (AGLTLVL, AGLTCODE);
Then organize the data into a useful hierarchy using a recursive query to assign group and row numbers within the structure:
CREATE TABLE #Data
(
grp integer NOT NULL,
rn integer NOT NULL,
ATYPCODE integer NOT NULL,
AGLTLVL integer NOT NULL,
AGLTCODE varchar(50) NOT NULL,
PRIMARY KEY (grp, rn DESC)
);
WITH R AS
(
-- Anchor: rows where AGLTLVL = 1
SELECT
grp = ROW_NUMBER() OVER (ORDER BY A.AGLTCODE),
rn = 1,
A.ATYPCODE,
A.AGLTLVL,
AGLTCODE = CONVERT(varchar(50), A.AGLTCODE + SPACE(2))
FROM dbo.ACCOUNT2 AS A
WHERE
A.AGLTLVL = 1
UNION ALL
-- Recursive: find the next AGLTLVL row in sequence
SELECT
R.grp,
R.rn + 1,
A.ATYPCODE,
A.AGLTLVL,
A.AGLTCODE
FROM R
JOIN dbo.ACCOUNT2 AS A WITH (FORCESEEK)
ON A.AGLTLVL = R.AGLTLVL + 1
AND A.AGLTCODE LIKE R.AGLTCODE + '%'
AND A.AGLTCODE = R.AGLTCODE + CONVERT(varchar(11), A.ATYPCODE)
)
INSERT #Data
(
grp,
rn,
ATYPCODE,
AGLTLVL,
AGLTCODE
)
SELECT
R.grp,
R.rn,
R.ATYPCODE,
R.AGLTLVL,
R.AGLTCODE
FROM R
OPTION (MAXRECURSION 0);
The contents of the #Data
table at this point are:
╔═════╦════╦══════════╦═════════╦══════════════════╗
║ grp ║ rn ║ ATYPCODE ║ AGLTLVL ║ AGLTCODE ║
╠═════╬════╬══════════╬═════════╬══════════════════╣
║ 1 ║ 1 ║ 3000 ║ 1 ║ 3000 ║
║ 1 ║ 2 ║ 19 ║ 2 ║ 3000 19 ║
║ 1 ║ 3 ║ 1170 ║ 3 ║ 3000 191170 ║
║ 1 ║ 4 ║ 1178 ║ 4 ║ 3000 1911701178 ║
║ 2 ║ 1 ║ 4000 ║ 1 ║ 4000 ║
║ 2 ║ 2 ║ 18 ║ 2 ║ 4000 18 ║
║ 2 ║ 3 ║ 2020 ║ 3 ║ 4000 182020 ║
╚═════╩════╩══════════╩═════════╩══════════════════╝
Then the final query becomes much easier. We take the values from the highest row number per group for some values, and pivot the others, while adding in the type descriptions from the other table:
SELECT
SQ2.ATYPCODE,
SQ2.AGLTCODE,
SQ2.AGLTLVL,
SQ2.ATYPCODE1,
-- Fill in any missing type codes
ATYPCODE2 = COALESCE(SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE3 = COALESCE(SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE4 = COALESCE(SQ2.ATYPCODE4, SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
-- Fill in any missing type descriptions
ATYPDESC2 = COALESCE(SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC3 = COALESCE(SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC4 = COALESCE(SQ2.ATYPDESC4, SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1)
FROM
(
SELECT
-- Same in every row anyway
ATYPCODE = MAX(SQ1.ATYPCODE),
AGLTCODE = MAX(SQ1.AGLTCODE),
AGLTLVL = MAX(SQ1.AGLTLVL),
-- Pivot type codes
ATYPCODE1 = MAX(IIF(SQ1.rn = 1, SQ1.TypeCode, NULL)),
ATYPCODE2 = MAX(IIF(SQ1.rn = 2, SQ1.TypeCode, NULL)),
ATYPCODE3 = MAX(IIF(SQ1.rn = 3, SQ1.TypeCode, NULL)),
ATYPCODE4 = MAX(IIF(SQ1.rn = 4, SQ1.TypeCode, NULL)),
-- Pivot type descriptions
ATYPDESC1 = MAX(IIF(SQ1.rn = 1, SQ1.ATYPDESC, NULL)),
ATYPDESC2 = MAX(IIF(SQ1.rn = 2, SQ1.ATYPDESC, NULL)),
ATYPDESC3 = MAX(IIF(SQ1.rn = 3, SQ1.ATYPDESC, NULL)),
ATYPDESC4 = MAX(IIF(SQ1.rn = 4, SQ1.ATYPDESC, NULL))
FROM
(
SELECT
-- Values taken from highest row number per group
ATYPCODE = FIRST_VALUE(D.ATYPCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTCODE = FIRST_VALUE(D.AGLTCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTLVL = FIRST_VALUE(D.AGLTLVL) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
-- Pivot data
TypeCode = D.ATYPCODE,
A.ATYPDESC,
-- Groups and row numbers
D.grp,
D.rn
FROM #Data AS D
JOIN dbo.ACCOUNT1 AS A
ON A.ATYPCODE = D.ATYPCODE
) AS SQ1
GROUP BY
SQ1.grp
) AS SQ2;
db<>fiddle demo
╔══════════╦══════════════════╦═════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╗
║ ATYPCODE ║ AGLTCODE ║ AGLTLVL ║ ATYPCODE1 ║ ATYPCODE2 ║ ATYPCODE3 ║ ATYPCODE4 ║ ATYPDESC2 ║ ATYPDESC3 ║ ATYPDESC4 ║
╠══════════╬══════════════════╬═════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╣
║ 1178 ║ 3000 1911701178 ║ 4 ║ 3000 ║ 19 ║ 1170 ║ 1178 ║ B ║ C ║ D ║
║ 2020 ║ 4000 182020 ║ 3 ║ 4000 ║ 18 ║ 2020 ║ 2020 ║ BB ║ CC ║ CC ║
╚══════════╩══════════════════╩═════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╝
add a comment |
Your Answer
StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "182"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);
StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});
function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});
}
});
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fdba.stackexchange.com%2fquestions%2f242254%2fseparate-a-column-into-its-components-based-on-another-table%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
1 Answer
1
active
oldest
votes
1 Answer
1
active
oldest
votes
active
oldest
votes
active
oldest
votes
The first thing I would do is create some keys on the source tables:
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT1 (ATYPCODE);
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT2 (AGLTLVL, AGLTCODE);
Then organize the data into a useful hierarchy using a recursive query to assign group and row numbers within the structure:
CREATE TABLE #Data
(
grp integer NOT NULL,
rn integer NOT NULL,
ATYPCODE integer NOT NULL,
AGLTLVL integer NOT NULL,
AGLTCODE varchar(50) NOT NULL,
PRIMARY KEY (grp, rn DESC)
);
WITH R AS
(
-- Anchor: rows where AGLTLVL = 1
SELECT
grp = ROW_NUMBER() OVER (ORDER BY A.AGLTCODE),
rn = 1,
A.ATYPCODE,
A.AGLTLVL,
AGLTCODE = CONVERT(varchar(50), A.AGLTCODE + SPACE(2))
FROM dbo.ACCOUNT2 AS A
WHERE
A.AGLTLVL = 1
UNION ALL
-- Recursive: find the next AGLTLVL row in sequence
SELECT
R.grp,
R.rn + 1,
A.ATYPCODE,
A.AGLTLVL,
A.AGLTCODE
FROM R
JOIN dbo.ACCOUNT2 AS A WITH (FORCESEEK)
ON A.AGLTLVL = R.AGLTLVL + 1
AND A.AGLTCODE LIKE R.AGLTCODE + '%'
AND A.AGLTCODE = R.AGLTCODE + CONVERT(varchar(11), A.ATYPCODE)
)
INSERT #Data
(
grp,
rn,
ATYPCODE,
AGLTLVL,
AGLTCODE
)
SELECT
R.grp,
R.rn,
R.ATYPCODE,
R.AGLTLVL,
R.AGLTCODE
FROM R
OPTION (MAXRECURSION 0);
The contents of the #Data
table at this point are:
╔═════╦════╦══════════╦═════════╦══════════════════╗
║ grp ║ rn ║ ATYPCODE ║ AGLTLVL ║ AGLTCODE ║
╠═════╬════╬══════════╬═════════╬══════════════════╣
║ 1 ║ 1 ║ 3000 ║ 1 ║ 3000 ║
║ 1 ║ 2 ║ 19 ║ 2 ║ 3000 19 ║
║ 1 ║ 3 ║ 1170 ║ 3 ║ 3000 191170 ║
║ 1 ║ 4 ║ 1178 ║ 4 ║ 3000 1911701178 ║
║ 2 ║ 1 ║ 4000 ║ 1 ║ 4000 ║
║ 2 ║ 2 ║ 18 ║ 2 ║ 4000 18 ║
║ 2 ║ 3 ║ 2020 ║ 3 ║ 4000 182020 ║
╚═════╩════╩══════════╩═════════╩══════════════════╝
Then the final query becomes much easier. We take the values from the highest row number per group for some values, and pivot the others, while adding in the type descriptions from the other table:
SELECT
SQ2.ATYPCODE,
SQ2.AGLTCODE,
SQ2.AGLTLVL,
SQ2.ATYPCODE1,
-- Fill in any missing type codes
ATYPCODE2 = COALESCE(SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE3 = COALESCE(SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE4 = COALESCE(SQ2.ATYPCODE4, SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
-- Fill in any missing type descriptions
ATYPDESC2 = COALESCE(SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC3 = COALESCE(SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC4 = COALESCE(SQ2.ATYPDESC4, SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1)
FROM
(
SELECT
-- Same in every row anyway
ATYPCODE = MAX(SQ1.ATYPCODE),
AGLTCODE = MAX(SQ1.AGLTCODE),
AGLTLVL = MAX(SQ1.AGLTLVL),
-- Pivot type codes
ATYPCODE1 = MAX(IIF(SQ1.rn = 1, SQ1.TypeCode, NULL)),
ATYPCODE2 = MAX(IIF(SQ1.rn = 2, SQ1.TypeCode, NULL)),
ATYPCODE3 = MAX(IIF(SQ1.rn = 3, SQ1.TypeCode, NULL)),
ATYPCODE4 = MAX(IIF(SQ1.rn = 4, SQ1.TypeCode, NULL)),
-- Pivot type descriptions
ATYPDESC1 = MAX(IIF(SQ1.rn = 1, SQ1.ATYPDESC, NULL)),
ATYPDESC2 = MAX(IIF(SQ1.rn = 2, SQ1.ATYPDESC, NULL)),
ATYPDESC3 = MAX(IIF(SQ1.rn = 3, SQ1.ATYPDESC, NULL)),
ATYPDESC4 = MAX(IIF(SQ1.rn = 4, SQ1.ATYPDESC, NULL))
FROM
(
SELECT
-- Values taken from highest row number per group
ATYPCODE = FIRST_VALUE(D.ATYPCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTCODE = FIRST_VALUE(D.AGLTCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTLVL = FIRST_VALUE(D.AGLTLVL) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
-- Pivot data
TypeCode = D.ATYPCODE,
A.ATYPDESC,
-- Groups and row numbers
D.grp,
D.rn
FROM #Data AS D
JOIN dbo.ACCOUNT1 AS A
ON A.ATYPCODE = D.ATYPCODE
) AS SQ1
GROUP BY
SQ1.grp
) AS SQ2;
db<>fiddle demo
╔══════════╦══════════════════╦═════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╗
║ ATYPCODE ║ AGLTCODE ║ AGLTLVL ║ ATYPCODE1 ║ ATYPCODE2 ║ ATYPCODE3 ║ ATYPCODE4 ║ ATYPDESC2 ║ ATYPDESC3 ║ ATYPDESC4 ║
╠══════════╬══════════════════╬═════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╣
║ 1178 ║ 3000 1911701178 ║ 4 ║ 3000 ║ 19 ║ 1170 ║ 1178 ║ B ║ C ║ D ║
║ 2020 ║ 4000 182020 ║ 3 ║ 4000 ║ 18 ║ 2020 ║ 2020 ║ BB ║ CC ║ CC ║
╚══════════╩══════════════════╩═════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╝
add a comment |
The first thing I would do is create some keys on the source tables:
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT1 (ATYPCODE);
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT2 (AGLTLVL, AGLTCODE);
Then organize the data into a useful hierarchy using a recursive query to assign group and row numbers within the structure:
CREATE TABLE #Data
(
grp integer NOT NULL,
rn integer NOT NULL,
ATYPCODE integer NOT NULL,
AGLTLVL integer NOT NULL,
AGLTCODE varchar(50) NOT NULL,
PRIMARY KEY (grp, rn DESC)
);
WITH R AS
(
-- Anchor: rows where AGLTLVL = 1
SELECT
grp = ROW_NUMBER() OVER (ORDER BY A.AGLTCODE),
rn = 1,
A.ATYPCODE,
A.AGLTLVL,
AGLTCODE = CONVERT(varchar(50), A.AGLTCODE + SPACE(2))
FROM dbo.ACCOUNT2 AS A
WHERE
A.AGLTLVL = 1
UNION ALL
-- Recursive: find the next AGLTLVL row in sequence
SELECT
R.grp,
R.rn + 1,
A.ATYPCODE,
A.AGLTLVL,
A.AGLTCODE
FROM R
JOIN dbo.ACCOUNT2 AS A WITH (FORCESEEK)
ON A.AGLTLVL = R.AGLTLVL + 1
AND A.AGLTCODE LIKE R.AGLTCODE + '%'
AND A.AGLTCODE = R.AGLTCODE + CONVERT(varchar(11), A.ATYPCODE)
)
INSERT #Data
(
grp,
rn,
ATYPCODE,
AGLTLVL,
AGLTCODE
)
SELECT
R.grp,
R.rn,
R.ATYPCODE,
R.AGLTLVL,
R.AGLTCODE
FROM R
OPTION (MAXRECURSION 0);
The contents of the #Data
table at this point are:
╔═════╦════╦══════════╦═════════╦══════════════════╗
║ grp ║ rn ║ ATYPCODE ║ AGLTLVL ║ AGLTCODE ║
╠═════╬════╬══════════╬═════════╬══════════════════╣
║ 1 ║ 1 ║ 3000 ║ 1 ║ 3000 ║
║ 1 ║ 2 ║ 19 ║ 2 ║ 3000 19 ║
║ 1 ║ 3 ║ 1170 ║ 3 ║ 3000 191170 ║
║ 1 ║ 4 ║ 1178 ║ 4 ║ 3000 1911701178 ║
║ 2 ║ 1 ║ 4000 ║ 1 ║ 4000 ║
║ 2 ║ 2 ║ 18 ║ 2 ║ 4000 18 ║
║ 2 ║ 3 ║ 2020 ║ 3 ║ 4000 182020 ║
╚═════╩════╩══════════╩═════════╩══════════════════╝
Then the final query becomes much easier. We take the values from the highest row number per group for some values, and pivot the others, while adding in the type descriptions from the other table:
SELECT
SQ2.ATYPCODE,
SQ2.AGLTCODE,
SQ2.AGLTLVL,
SQ2.ATYPCODE1,
-- Fill in any missing type codes
ATYPCODE2 = COALESCE(SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE3 = COALESCE(SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE4 = COALESCE(SQ2.ATYPCODE4, SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
-- Fill in any missing type descriptions
ATYPDESC2 = COALESCE(SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC3 = COALESCE(SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC4 = COALESCE(SQ2.ATYPDESC4, SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1)
FROM
(
SELECT
-- Same in every row anyway
ATYPCODE = MAX(SQ1.ATYPCODE),
AGLTCODE = MAX(SQ1.AGLTCODE),
AGLTLVL = MAX(SQ1.AGLTLVL),
-- Pivot type codes
ATYPCODE1 = MAX(IIF(SQ1.rn = 1, SQ1.TypeCode, NULL)),
ATYPCODE2 = MAX(IIF(SQ1.rn = 2, SQ1.TypeCode, NULL)),
ATYPCODE3 = MAX(IIF(SQ1.rn = 3, SQ1.TypeCode, NULL)),
ATYPCODE4 = MAX(IIF(SQ1.rn = 4, SQ1.TypeCode, NULL)),
-- Pivot type descriptions
ATYPDESC1 = MAX(IIF(SQ1.rn = 1, SQ1.ATYPDESC, NULL)),
ATYPDESC2 = MAX(IIF(SQ1.rn = 2, SQ1.ATYPDESC, NULL)),
ATYPDESC3 = MAX(IIF(SQ1.rn = 3, SQ1.ATYPDESC, NULL)),
ATYPDESC4 = MAX(IIF(SQ1.rn = 4, SQ1.ATYPDESC, NULL))
FROM
(
SELECT
-- Values taken from highest row number per group
ATYPCODE = FIRST_VALUE(D.ATYPCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTCODE = FIRST_VALUE(D.AGLTCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTLVL = FIRST_VALUE(D.AGLTLVL) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
-- Pivot data
TypeCode = D.ATYPCODE,
A.ATYPDESC,
-- Groups and row numbers
D.grp,
D.rn
FROM #Data AS D
JOIN dbo.ACCOUNT1 AS A
ON A.ATYPCODE = D.ATYPCODE
) AS SQ1
GROUP BY
SQ1.grp
) AS SQ2;
db<>fiddle demo
╔══════════╦══════════════════╦═════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╗
║ ATYPCODE ║ AGLTCODE ║ AGLTLVL ║ ATYPCODE1 ║ ATYPCODE2 ║ ATYPCODE3 ║ ATYPCODE4 ║ ATYPDESC2 ║ ATYPDESC3 ║ ATYPDESC4 ║
╠══════════╬══════════════════╬═════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╣
║ 1178 ║ 3000 1911701178 ║ 4 ║ 3000 ║ 19 ║ 1170 ║ 1178 ║ B ║ C ║ D ║
║ 2020 ║ 4000 182020 ║ 3 ║ 4000 ║ 18 ║ 2020 ║ 2020 ║ BB ║ CC ║ CC ║
╚══════════╩══════════════════╩═════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╝
add a comment |
The first thing I would do is create some keys on the source tables:
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT1 (ATYPCODE);
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT2 (AGLTLVL, AGLTCODE);
Then organize the data into a useful hierarchy using a recursive query to assign group and row numbers within the structure:
CREATE TABLE #Data
(
grp integer NOT NULL,
rn integer NOT NULL,
ATYPCODE integer NOT NULL,
AGLTLVL integer NOT NULL,
AGLTCODE varchar(50) NOT NULL,
PRIMARY KEY (grp, rn DESC)
);
WITH R AS
(
-- Anchor: rows where AGLTLVL = 1
SELECT
grp = ROW_NUMBER() OVER (ORDER BY A.AGLTCODE),
rn = 1,
A.ATYPCODE,
A.AGLTLVL,
AGLTCODE = CONVERT(varchar(50), A.AGLTCODE + SPACE(2))
FROM dbo.ACCOUNT2 AS A
WHERE
A.AGLTLVL = 1
UNION ALL
-- Recursive: find the next AGLTLVL row in sequence
SELECT
R.grp,
R.rn + 1,
A.ATYPCODE,
A.AGLTLVL,
A.AGLTCODE
FROM R
JOIN dbo.ACCOUNT2 AS A WITH (FORCESEEK)
ON A.AGLTLVL = R.AGLTLVL + 1
AND A.AGLTCODE LIKE R.AGLTCODE + '%'
AND A.AGLTCODE = R.AGLTCODE + CONVERT(varchar(11), A.ATYPCODE)
)
INSERT #Data
(
grp,
rn,
ATYPCODE,
AGLTLVL,
AGLTCODE
)
SELECT
R.grp,
R.rn,
R.ATYPCODE,
R.AGLTLVL,
R.AGLTCODE
FROM R
OPTION (MAXRECURSION 0);
The contents of the #Data
table at this point are:
╔═════╦════╦══════════╦═════════╦══════════════════╗
║ grp ║ rn ║ ATYPCODE ║ AGLTLVL ║ AGLTCODE ║
╠═════╬════╬══════════╬═════════╬══════════════════╣
║ 1 ║ 1 ║ 3000 ║ 1 ║ 3000 ║
║ 1 ║ 2 ║ 19 ║ 2 ║ 3000 19 ║
║ 1 ║ 3 ║ 1170 ║ 3 ║ 3000 191170 ║
║ 1 ║ 4 ║ 1178 ║ 4 ║ 3000 1911701178 ║
║ 2 ║ 1 ║ 4000 ║ 1 ║ 4000 ║
║ 2 ║ 2 ║ 18 ║ 2 ║ 4000 18 ║
║ 2 ║ 3 ║ 2020 ║ 3 ║ 4000 182020 ║
╚═════╩════╩══════════╩═════════╩══════════════════╝
Then the final query becomes much easier. We take the values from the highest row number per group for some values, and pivot the others, while adding in the type descriptions from the other table:
SELECT
SQ2.ATYPCODE,
SQ2.AGLTCODE,
SQ2.AGLTLVL,
SQ2.ATYPCODE1,
-- Fill in any missing type codes
ATYPCODE2 = COALESCE(SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE3 = COALESCE(SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE4 = COALESCE(SQ2.ATYPCODE4, SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
-- Fill in any missing type descriptions
ATYPDESC2 = COALESCE(SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC3 = COALESCE(SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC4 = COALESCE(SQ2.ATYPDESC4, SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1)
FROM
(
SELECT
-- Same in every row anyway
ATYPCODE = MAX(SQ1.ATYPCODE),
AGLTCODE = MAX(SQ1.AGLTCODE),
AGLTLVL = MAX(SQ1.AGLTLVL),
-- Pivot type codes
ATYPCODE1 = MAX(IIF(SQ1.rn = 1, SQ1.TypeCode, NULL)),
ATYPCODE2 = MAX(IIF(SQ1.rn = 2, SQ1.TypeCode, NULL)),
ATYPCODE3 = MAX(IIF(SQ1.rn = 3, SQ1.TypeCode, NULL)),
ATYPCODE4 = MAX(IIF(SQ1.rn = 4, SQ1.TypeCode, NULL)),
-- Pivot type descriptions
ATYPDESC1 = MAX(IIF(SQ1.rn = 1, SQ1.ATYPDESC, NULL)),
ATYPDESC2 = MAX(IIF(SQ1.rn = 2, SQ1.ATYPDESC, NULL)),
ATYPDESC3 = MAX(IIF(SQ1.rn = 3, SQ1.ATYPDESC, NULL)),
ATYPDESC4 = MAX(IIF(SQ1.rn = 4, SQ1.ATYPDESC, NULL))
FROM
(
SELECT
-- Values taken from highest row number per group
ATYPCODE = FIRST_VALUE(D.ATYPCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTCODE = FIRST_VALUE(D.AGLTCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTLVL = FIRST_VALUE(D.AGLTLVL) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
-- Pivot data
TypeCode = D.ATYPCODE,
A.ATYPDESC,
-- Groups and row numbers
D.grp,
D.rn
FROM #Data AS D
JOIN dbo.ACCOUNT1 AS A
ON A.ATYPCODE = D.ATYPCODE
) AS SQ1
GROUP BY
SQ1.grp
) AS SQ2;
db<>fiddle demo
╔══════════╦══════════════════╦═════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╗
║ ATYPCODE ║ AGLTCODE ║ AGLTLVL ║ ATYPCODE1 ║ ATYPCODE2 ║ ATYPCODE3 ║ ATYPCODE4 ║ ATYPDESC2 ║ ATYPDESC3 ║ ATYPDESC4 ║
╠══════════╬══════════════════╬═════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╣
║ 1178 ║ 3000 1911701178 ║ 4 ║ 3000 ║ 19 ║ 1170 ║ 1178 ║ B ║ C ║ D ║
║ 2020 ║ 4000 182020 ║ 3 ║ 4000 ║ 18 ║ 2020 ║ 2020 ║ BB ║ CC ║ CC ║
╚══════════╩══════════════════╩═════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╝
The first thing I would do is create some keys on the source tables:
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT1 (ATYPCODE);
CREATE UNIQUE CLUSTERED INDEX cuq ON dbo.ACCOUNT2 (AGLTLVL, AGLTCODE);
Then organize the data into a useful hierarchy using a recursive query to assign group and row numbers within the structure:
CREATE TABLE #Data
(
grp integer NOT NULL,
rn integer NOT NULL,
ATYPCODE integer NOT NULL,
AGLTLVL integer NOT NULL,
AGLTCODE varchar(50) NOT NULL,
PRIMARY KEY (grp, rn DESC)
);
WITH R AS
(
-- Anchor: rows where AGLTLVL = 1
SELECT
grp = ROW_NUMBER() OVER (ORDER BY A.AGLTCODE),
rn = 1,
A.ATYPCODE,
A.AGLTLVL,
AGLTCODE = CONVERT(varchar(50), A.AGLTCODE + SPACE(2))
FROM dbo.ACCOUNT2 AS A
WHERE
A.AGLTLVL = 1
UNION ALL
-- Recursive: find the next AGLTLVL row in sequence
SELECT
R.grp,
R.rn + 1,
A.ATYPCODE,
A.AGLTLVL,
A.AGLTCODE
FROM R
JOIN dbo.ACCOUNT2 AS A WITH (FORCESEEK)
ON A.AGLTLVL = R.AGLTLVL + 1
AND A.AGLTCODE LIKE R.AGLTCODE + '%'
AND A.AGLTCODE = R.AGLTCODE + CONVERT(varchar(11), A.ATYPCODE)
)
INSERT #Data
(
grp,
rn,
ATYPCODE,
AGLTLVL,
AGLTCODE
)
SELECT
R.grp,
R.rn,
R.ATYPCODE,
R.AGLTLVL,
R.AGLTCODE
FROM R
OPTION (MAXRECURSION 0);
The contents of the #Data
table at this point are:
╔═════╦════╦══════════╦═════════╦══════════════════╗
║ grp ║ rn ║ ATYPCODE ║ AGLTLVL ║ AGLTCODE ║
╠═════╬════╬══════════╬═════════╬══════════════════╣
║ 1 ║ 1 ║ 3000 ║ 1 ║ 3000 ║
║ 1 ║ 2 ║ 19 ║ 2 ║ 3000 19 ║
║ 1 ║ 3 ║ 1170 ║ 3 ║ 3000 191170 ║
║ 1 ║ 4 ║ 1178 ║ 4 ║ 3000 1911701178 ║
║ 2 ║ 1 ║ 4000 ║ 1 ║ 4000 ║
║ 2 ║ 2 ║ 18 ║ 2 ║ 4000 18 ║
║ 2 ║ 3 ║ 2020 ║ 3 ║ 4000 182020 ║
╚═════╩════╩══════════╩═════════╩══════════════════╝
Then the final query becomes much easier. We take the values from the highest row number per group for some values, and pivot the others, while adding in the type descriptions from the other table:
SELECT
SQ2.ATYPCODE,
SQ2.AGLTCODE,
SQ2.AGLTLVL,
SQ2.ATYPCODE1,
-- Fill in any missing type codes
ATYPCODE2 = COALESCE(SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE3 = COALESCE(SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
ATYPCODE4 = COALESCE(SQ2.ATYPCODE4, SQ2.ATYPCODE3, SQ2.ATYPCODE2, SQ2.ATYPCODE1),
-- Fill in any missing type descriptions
ATYPDESC2 = COALESCE(SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC3 = COALESCE(SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1),
ATYPDESC4 = COALESCE(SQ2.ATYPDESC4, SQ2.ATYPDESC3, SQ2.ATYPDESC2, SQ2.ATYPDESC1)
FROM
(
SELECT
-- Same in every row anyway
ATYPCODE = MAX(SQ1.ATYPCODE),
AGLTCODE = MAX(SQ1.AGLTCODE),
AGLTLVL = MAX(SQ1.AGLTLVL),
-- Pivot type codes
ATYPCODE1 = MAX(IIF(SQ1.rn = 1, SQ1.TypeCode, NULL)),
ATYPCODE2 = MAX(IIF(SQ1.rn = 2, SQ1.TypeCode, NULL)),
ATYPCODE3 = MAX(IIF(SQ1.rn = 3, SQ1.TypeCode, NULL)),
ATYPCODE4 = MAX(IIF(SQ1.rn = 4, SQ1.TypeCode, NULL)),
-- Pivot type descriptions
ATYPDESC1 = MAX(IIF(SQ1.rn = 1, SQ1.ATYPDESC, NULL)),
ATYPDESC2 = MAX(IIF(SQ1.rn = 2, SQ1.ATYPDESC, NULL)),
ATYPDESC3 = MAX(IIF(SQ1.rn = 3, SQ1.ATYPDESC, NULL)),
ATYPDESC4 = MAX(IIF(SQ1.rn = 4, SQ1.ATYPDESC, NULL))
FROM
(
SELECT
-- Values taken from highest row number per group
ATYPCODE = FIRST_VALUE(D.ATYPCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTCODE = FIRST_VALUE(D.AGLTCODE) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
AGLTLVL = FIRST_VALUE(D.AGLTLVL) OVER (
PARTITION BY D.grp
ORDER BY D.rn DESC
ROWS UNBOUNDED PRECEDING),
-- Pivot data
TypeCode = D.ATYPCODE,
A.ATYPDESC,
-- Groups and row numbers
D.grp,
D.rn
FROM #Data AS D
JOIN dbo.ACCOUNT1 AS A
ON A.ATYPCODE = D.ATYPCODE
) AS SQ1
GROUP BY
SQ1.grp
) AS SQ2;
db<>fiddle demo
╔══════════╦══════════════════╦═════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╦═══════════╗
║ ATYPCODE ║ AGLTCODE ║ AGLTLVL ║ ATYPCODE1 ║ ATYPCODE2 ║ ATYPCODE3 ║ ATYPCODE4 ║ ATYPDESC2 ║ ATYPDESC3 ║ ATYPDESC4 ║
╠══════════╬══════════════════╬═════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╬═══════════╣
║ 1178 ║ 3000 1911701178 ║ 4 ║ 3000 ║ 19 ║ 1170 ║ 1178 ║ B ║ C ║ D ║
║ 2020 ║ 4000 182020 ║ 3 ║ 4000 ║ 18 ║ 2020 ║ 2020 ║ BB ║ CC ║ CC ║
╚══════════╩══════════════════╩═════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╩═══════════╝
edited 6 hours ago
answered 9 hours ago
Paul White♦Paul White
57.4k15 gold badges302 silver badges475 bronze badges
57.4k15 gold badges302 silver badges475 bronze badges
add a comment |
add a comment |
Thanks for contributing an answer to Database Administrators Stack Exchange!
- Please be sure to answer the question. Provide details and share your research!
But avoid …
- Asking for help, clarification, or responding to other answers.
- Making statements based on opinion; back them up with references or personal experience.
To learn more, see our tips on writing great answers.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fdba.stackexchange.com%2fquestions%2f242254%2fseparate-a-column-into-its-components-based-on-another-table%23new-answer', 'question_page');
}
);
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown
Required, but never shown