এই সমস্যাটি সমাধান করার জন্য একটি স্বজ্ঞাত পদ্ধতি:
- প্রতিটি দলের জন্য সর্বাধিক সাম্প্রতিক ফলাফল সন্ধান করুন
- পূর্ববর্তী ম্যাচটি পরীক্ষা করে দেখুন এবং ফলাফলের ধরণটি মিলে গেলে স্ট্রাইক কাউন্টে একটি যুক্ত করুন
- দ্বিতীয় ধাপটি পুনরাবৃত্তি করুন তবে প্রথম আলাদা ফলাফলের সাথে সাথেই থামুন
এই কৌশলটি উইন্ডো ফাংশন সলিউশন (যা ডেটার পুরো স্ক্যান সম্পাদন করে) জেতে পারে, কারণ টেবিলটি বড় হতে থাকে, ধরে নিচ্ছে পুনরাবৃত্ত কৌশলটি দক্ষতার সাথে প্রয়োগ করা হয়েছে। সাফল্যের মূল চাবিকাঠি হ'ল সারিগুলি দ্রুত সন্ধান করতে দক্ষতার সূচক সরবরাহ করা (সিক্স ব্যবহার করে) এবং প্রকারগুলি এড়ানো। সূচকগুলি প্রয়োজনীয়:
-- New index #1
CREATE UNIQUE INDEX uq1 ON dbo.FantasyMatches
(home_fantasy_team_id, match_id)
INCLUDE (winning_team_id);
-- New index #2
CREATE UNIQUE INDEX uq2 ON dbo.FantasyMatches
(away_fantasy_team_id, match_id)
INCLUDE (winning_team_id);
ক্যোয়ারী অপ্টিমাইজেশনে সহায়তা করার জন্য, আমি একটি বর্তমান অলঙ্কারের অংশ হিসাবে চিহ্নিত সারিগুলি ধরে রাখতে একটি অস্থায়ী টেবিল ব্যবহার করব। যদি রেখাগুলি সাধারণত ছোট হয় (আমি যে দলগুলি অনুসরণ করি তাদের পক্ষে সত্য, দুঃখের সাথে) এই টেবিলটি বেশ ছোট হওয়া উচিত:
-- Table to hold just the rows that form streaks
CREATE TABLE #StreakData
(
team_id bigint NOT NULL,
match_id bigint NOT NULL,
streak_type char(1) NOT NULL,
streak_length integer NOT NULL,
);
-- Temporary table unique clustered index
CREATE UNIQUE CLUSTERED INDEX cuq ON #StreakData (team_id, match_id);
আমার পুনরাবৃত্তির ক্যোয়ারী সমাধানটি নীচে রয়েছে ( এসকিউএল ফিডল এখানে ):
-- Solution query
WITH Streaks AS
(
-- Anchor: most recent match for each team
SELECT
FT.team_id,
CA.match_id,
CA.streak_type,
streak_length = 1
FROM dbo.FantasyTeams AS FT
CROSS APPLY
(
-- Most recent match
SELECT
T.match_id,
T.streak_type
FROM
(
SELECT
FM.match_id,
streak_type =
CASE
WHEN FM.winning_team_id = FM.home_fantasy_team_id
THEN CONVERT(char(1), 'W')
WHEN FM.winning_team_id IS NULL
THEN CONVERT(char(1), 'T')
ELSE CONVERT(char(1), 'L')
END
FROM dbo.FantasyMatches AS FM
WHERE
FT.team_id = FM.home_fantasy_team_id
UNION ALL
SELECT
FM.match_id,
streak_type =
CASE
WHEN FM.winning_team_id = FM.away_fantasy_team_id
THEN CONVERT(char(1), 'W')
WHEN FM.winning_team_id IS NULL
THEN CONVERT(char(1), 'T')
ELSE CONVERT(char(1), 'L')
END
FROM dbo.FantasyMatches AS FM
WHERE
FT.team_id = FM.away_fantasy_team_id
) AS T
ORDER BY
T.match_id DESC
OFFSET 0 ROWS
FETCH FIRST 1 ROW ONLY
) AS CA
UNION ALL
-- Recursive part: prior match with the same streak type
SELECT
Streaks.team_id,
LastMatch.match_id,
Streaks.streak_type,
Streaks.streak_length + 1
FROM Streaks
CROSS APPLY
(
-- Most recent prior match
SELECT
Numbered.match_id,
Numbered.winning_team_id,
Numbered.team_id
FROM
(
-- Assign a row number
SELECT
PreviousMatches.match_id,
PreviousMatches.winning_team_id,
PreviousMatches.team_id,
rn = ROW_NUMBER() OVER (
ORDER BY PreviousMatches.match_id DESC)
FROM
(
-- Prior match as home or away team
SELECT
FM.match_id,
FM.winning_team_id,
team_id = FM.home_fantasy_team_id
FROM dbo.FantasyMatches AS FM
WHERE
FM.home_fantasy_team_id = Streaks.team_id
AND FM.match_id < Streaks.match_id
UNION ALL
SELECT
FM.match_id,
FM.winning_team_id,
team_id = FM.away_fantasy_team_id
FROM dbo.FantasyMatches AS FM
WHERE
FM.away_fantasy_team_id = Streaks.team_id
AND FM.match_id < Streaks.match_id
) AS PreviousMatches
) AS Numbered
-- Most recent
WHERE
Numbered.rn = 1
) AS LastMatch
-- Check the streak type matches
WHERE EXISTS
(
SELECT
Streaks.streak_type
INTERSECT
SELECT
CASE
WHEN LastMatch.winning_team_id IS NULL THEN 'T'
WHEN LastMatch.winning_team_id = LastMatch.team_id THEN 'W'
ELSE 'L'
END
)
)
INSERT #StreakData
(team_id, match_id, streak_type, streak_length)
SELECT
team_id,
match_id,
streak_type,
streak_length
FROM Streaks
OPTION (MAXRECURSION 0);
টি-এসকিউএল পাঠ্যটি বেশ দীর্ঘ, তবে ক্যোয়ারীর প্রতিটি বিভাগ এই উত্তরের শুরুতে প্রদত্ত বিস্তৃত প্রক্রিয়ার রূপরেখার সাথে খুব ঘনিষ্ঠভাবে মিলছে। প্রকারগুলি এড়াতে এবং TOP
কোয়েরির পুনরাবৃত্তির অংশে (যা সাধারণত অনুমোদিত নয়) উত্পাদনের জন্য নির্দিষ্ট কৌশলগুলি ব্যবহার করার প্রয়োজনে ক্যোরিয়াকে দীর্ঘ করা হয় ।
পরীক্ষার পরিকল্পনাটি কোয়েরির সাথে তুলনা করে তুলনামূলকভাবে ছোট এবং সহজ। আমি অ্যাঙ্কার অঞ্চলটি হলুদ করে দিয়েছি, এবং নীচের স্ক্রিনশটটিতে পুনরাবৃত্ত অংশটি সবুজ করেছে:
অস্থায়ী টেবিলের মধ্যে থাকা স্ট্রাইক সারিগুলির সাথে, আপনার প্রয়োজনীয় সংক্ষিপ্ত ফলাফলগুলি পাওয়া সহজ। (অস্থায়ী টেবিল ব্যবহার করে বাছাইয়ের স্পিল এড়ানো যায় যা নীচের কোয়েরিটি মূল পুনরাবৃত্ত ক্যোয়ারীর সাথে মিলিত হলে ঘটতে পারে)
-- Basic results
SELECT
SD.team_id,
StreakType = MAX(SD.streak_type),
StreakLength = MAX(SD.streak_length)
FROM #StreakData AS SD
GROUP BY
SD.team_id
ORDER BY
SD.team_id;
FantasyTeams
টেবিলটি আপডেট করার জন্য একই প্রশ্নটি ভিত্তি হিসাবে ব্যবহার করা যেতে পারে :
-- Update team summary
WITH StreakData AS
(
SELECT
SD.team_id,
StreakType = MAX(SD.streak_type),
StreakLength = MAX(SD.streak_length)
FROM #StreakData AS SD
GROUP BY
SD.team_id
)
UPDATE FT
SET streak_type = SD.StreakType,
streak_count = SD.StreakLength
FROM StreakData AS SD
JOIN dbo.FantasyTeams AS FT
ON FT.team_id = SD.team_id;
বা, যদি আপনি পছন্দ করেন MERGE
:
MERGE dbo.FantasyTeams AS FT
USING
(
SELECT
SD.team_id,
StreakType = MAX(SD.streak_type),
StreakLength = MAX(SD.streak_length)
FROM #StreakData AS SD
GROUP BY
SD.team_id
) AS StreakData
ON StreakData.team_id = FT.team_id
WHEN MATCHED THEN UPDATE SET
FT.streak_type = StreakData.StreakType,
FT.streak_count = StreakData.StreakLength;
হয় পদ্ধতির একটি কার্যকর সম্পাদন পরিকল্পনা উত্পাদন করে (অস্থায়ী সারণীতে সারিগুলির পরিচিত সংখ্যার ভিত্তিতে):
অবশেষে, কারণ পুনরাবৃত্তির পদ্ধতিটি প্রাকৃতিকভাবে match_id
এর প্রসেসিংয়ে অন্তর্ভুক্ত করে , সুতরাং match_id
প্রতিটি আউটপুটে প্রতিটি লাইন তৈরি করে এমন একটির তালিকা যুক্ত করা সহজ :
SELECT
S.team_id,
streak_type = MAX(S.streak_type),
match_id_list =
STUFF(
(
SELECT ',' + CONVERT(varchar(11), S2.match_id)
FROM #StreakData AS S2
WHERE S2.team_id = S.team_id
ORDER BY S2.match_id DESC
FOR XML PATH ('')
), 1, 1, ''),
streak_length = MAX(S.streak_length)
FROM #StreakData AS S
GROUP BY
S.team_id
ORDER BY
S.team_id;
আউটপুট:
হত্যা পরিকল্পনা: