SQL Server2008R2を使用しています。
このテーブル@t(ORDER BY PK DESCのトップ20)について考えてみます。
PK SK VC APP M C
== == == ==== == ==================
21 7 79 NULL 0 NULL
20 9 74 1 3 20=14, 18=13, 15=2
19 6 79 1 2 19=11, 17=7
18 9 77 1 0 NULL
17 6 74 1 0 NULL
16 7 79 1 0 NULL
15 9 74 1 0 NULL
14 9 74 1 0 NULL
13 9 77 1 0 NULL
12 7 77 1 0 NULL
11 6 79 1 0 NULL
10 7 79 1 0 NULL
9 7 74 1 0 NULL
8 7 79 1 0 NULL
7 6 74 1 0 NULL
6 6 74 1 0 NULL
5 7 79 1 0 NULL
4 7 77 1 0 NULL
3 6 79 1 0 NULL
2 9 74 1 0 NULL
これで作成:
DECLARE @t TABLE(PK INT NOT NULL IDENTITY(1,1) PRIMARY KEY CLUSTERED, SK INT NOT NULL, VC INT NULL, APP INT NULL, M INT NOT NULL, C NVARCHAR(111) NULL);
INSERT @t (SK,VC,APP,M,C) VALUES
(7,77,1,0,NULL),
(9,74,1,0,NULL),
(6,79,1,0,NULL),
(7,77,1,0,NULL),
(7,79,1,0,NULL),
(6,74,1,0,NULL),
(6,74,1,0,NULL),
(7,79,1,0,NULL),
(7,74,1,0,NULL),
(7,79,1,0,NULL),
(6,79,1,0,NULL),
(7,77,1,0,NULL),
(9,77,1,0,NULL),
(9,74,1,0,NULL),
(9,74,1,0,NULL),
(7,79,1,0,NULL),
(6,74,1,0,NULL),
(9,77,1,0,NULL),
(6,79,1,2,'19=11, 17=7'),
(9,74,1,3,'20=14, 18=13, 15=2'),
(7,79,NULL,0,NULL)
true
私のタスクは、最新の行(where APP IS NOT NULL
)が一連のX一致ペアまたは同じグループ(同じ現在のSK)の最新の行を完了した場合に一致に戻ることです。
たとえば、2ペアのみをテストする場合、現在必要なテストがSK = 6であるとすると、PK=19に到達するとすぐに一致があります。
一致はVC(19)= VC(11)= 79 AND VC(17)= VC(7)=74です。
以下を実行してそれを確認してください。
DECLARE @PairsToTest int = 2
DECLARE @SK int = 6
SELECT
TOP (2*@PairsToTest)
*
FROM @t
WHERE
APP IS NOT NULL
AND SK = @SK
ORDER BY SK, PK DESC
結果:
PK SK VC APP M C
19 6 79 1 2 19=11, 17=7
17 6 74 1 0 NULL
11 6 79 1 0 NULL
7 6 74 1 0 NULL
もう一つの例:
3ペアをテストする場合、SK=9を調べるとPK=20で一致が見つかります(それ自体は興味深い質問ですが、私のタスクではすべてのSKをテストする必要はありません。特定のSKの結果で十分です。私のため。
一致を確認するには、次のコマンドを実行します。
DECLARE @PairsToTest int = 3
DECLARE @SK int = 9
SELECT
TOP (2*@PairsToTest)
*
FROM @t
WHERE
APP IS NOT NULL
AND SK = @SK
ORDER BY SK, PK DESC
結果:
PK SK VC APP M C
20 9 74 1 3 20=14, 18=13, 15=2
18 9 77 1 0 NULL
15 9 74 1 0 NULL
14 9 74 1 0 NULL
13 9 77 1 0 NULL
2 9 74 1 0 NULL
ご覧のとおり、VC(20)= VC(14)= 74、VC(18)= VC(13)= 74、VC(15)= VC(2)
必要な行のセットを正しい順序で選択し、VCで等しい行を数えることを考えました。カウントが同じである場合、@PairsToTest
これはフラグを立てるためのサインです。
私は試した:
DECLARE @PairsToTest int = 3
DECLARE @SK int = 9
;with t0 as
(
SELECT
TOP (2*@PairsToTest)
*
FROM @t
WHERE
APP IS NOT NULL
AND SK = @SK
ORDER BY SK, PK DESC
),
t1 AS
(
SELECT TOP (@PairsToTest) * FROM t0
),
t2 AS
(
SELECT TOP (@PairsToTest) * FROM t0 ORDER BY PK ASC
)
,t3 AS
(
SELECT TOP 99999999 * FROM t2 ORDER BY PK DESC
)
IF (SELECT COUNT(*) FROM t1 LEFT OUTER JOIN t3 ON t1.VC = t3.VC) = @PairsToTest
SELECT 1
ELSE
SELECT 0
しかし、これにも欠陥があるかもしれません:
- VCには一意のデータが含まれていません(偶然のみ)
- IFは許可されていません
- 私はt3でTOP99999999を取り除く必要があります(私はそれと一緒に暮らすことができますが)
これを解決するために必要な変更は何ですか?