IIFの場合の交換の書き込みに関する問題によるWorking Access SQLクエリのT-SQL移行
質問
私は2つのテーブル bmreports_fpn_curves と bmreports_boa_curves の名前、日時、期間、および値で構成されています。
.
BM_UNIT_NAME RunDate Period FPN (or BOA)
T_DRAXX-1 2010-12-01 00:03:00 1 497
rundateフィールドは1分(1日あたりのレコード)と1~48の期間で増加します。 bmreports_fpn_curves iは、各期間ごとに完全なデータ・セットを持ち、 bmreports_boa_curves には、これらの基本値を置き換える値が含まれています。
通常重複するBOA値があり、アクセス SQLステートメントのネストされたIIFステートメントは、FPN、MAX BOA値またはMIN BOAの値のいずれかのいずれかのポイントのいずれかの1つを選択するためのルールを含んでいました。 。規則は次のとおりです。
1)BOA値がない場合は、FPN値を使用してください。
2)BOA値がある場合はFPNよりも小さい場合は、Min BOA値を見つけて使用する
3)BOA値がある場合はFPNより大きい場合は、MAX BOA値を見つけて使用してください。
アクセスSQLクエリは完全に機能し、次のとおりです。
.
SELECT
dbo_BMReports_FPN_Curves.BM_Unit_Name,
dbo_BMReports_FPN_Curves.RunDate,
dbo_BMReports_FPN_Curves.Period,
dbo_BMReports_FPN_Curves.PN_Level,
IIf(IIf(Min([dbo_BMReports_BOA_Curves]![PN_Level]) <[dbo_BMReports_FPN_Curves]![PN_Level],Min([dbo_BMReports_BOA_Curves]! [PN_Level]),Max([dbo_BMReports_BOA_Curves]![PN_Level])) Is Null, [dbo_BMReports_FPN_Curves]![PN_Level],
IIf(Min([dbo_BMReports_BOA_Curves]![PN_Level])<[dbo_BMReports_FPN_Curves]! [PN_Level],Min([dbo_BMReports_BOA_Curves]! [PN_Level]),Max([dbo_BMReports_BOA_Curves]![PN_Level]))) AS BOA
FROM dbo_BMReports_FPN_Curves LEFT JOIN dbo_BMReports_BOA_Curves ON (dbo_BMReports_FPN_Curves.RunDate = dbo_BMReports_BOA_Curves.RunDate) AND (dbo_BMReports_FPN_Curves.BM_Unit_Name = dbo_BMReports_BOA_Curves.BM_Unit_Name)
GROUP BY dbo_BMReports_FPN_Curves.BM_Unit_Name, dbo_BMReports_FPN_Curves.RunDate, dbo_BMReports_FPN_Curves.Period, dbo_BMReports_FPN_Curves.PN_Level
HAVING (((dbo_BMReports_FPN_Curves.BM_Unit_Name)='T_DRAXX-1'));
T-SQL(同じSQL Serverデータソースの問合せ)でクエリの大部分を書き直し、残りのグループを作成し、要素をすべて稼働させているが、交換時に停止しています。 IFFのために、いくつかの瞬間がある場合は手に感謝します。
現在スタンド化されているSQLクエリ:
.SELECT
BMReports_FPN_Curves.BM_Unit_Name,
BMReports_FPN_Curves.RunDate,
BMReports_FPN_Curves.Period,
AVG(BMReports_FPN_Curves.PN_Level) AS FPN,
CASE
WHEN BMReports_BOA_Curves.PN_Level IS NULL THEN AVG(BMReports_FPN_Curves.PN_Level)
WHEN MIN(BMReports_BOA_Curves.PN_Level) IS < AVG(BMReports_FPN_Curves.PN_Level) THEN MIN(BMReports_BOA_Curves.PN_Level)
ELSE MAX(BMReports_BOA_Curves.PN_Level)
END AS BOA
FROM BMReports_FPN_Curves
LEFT JOIN BMReports_BOA_Curves ON BMReports_FPN_Curves.BM_Unit_Name = BMReports_BOA_Curves.BM_Unit_Name
AND BMReports_FPN_Curves.RunDate = BMReports_BOA_Curves.RunDate
GROUP BY BMReports_FPN_Curves.BM_Unit_Name, BMReports_FPN_Curves.RunDate, BMReports_FPN_Curves.Period
HAVING BMReports_FPN_Curves.BM_Unit_Name = 'T_DRAXX-1'
ORDER BY BMReports_FPN_Curves.BM_Unit_Name, BMReports_FPN_Curves.RunDate, BMReports_FPN_Curves.Period
解決
SELECT fc.BM_Unit_Name
, fc.RunDate
, fc.Period
, CASE
WHEN AVG(bc.PN_Level) IS NULL THEN AVG(fc.PN_Level) -- No BOA Value, use the FPN Value
WHEN MIN(bc.PN_Level) < AVG(fc.PN_Level) THEN MIN(bc.PN_Level) -- BOA Value is less than the FPN, use the BOA Value
ELSE MAX(bc.PN_Level) -- BOA Value is greater than the FPN, use the BOA Value
END
FROM dbo.BMReports_FPN_Curves fc
LEFT JOIN dbo.BMReports_BOA_Curves bc ON fc.RunDate = bc.RunDate
AND fc.BM_Unit_Name = bc.BM_Unit_Name
WHERE fc.BM_Unit_Name ='T_DRAXX-1'
GROUP BY
fc.BM_Unit_Name
, fc.RunDate
, fc.Period
. 他のヒント
CTEを使用して、すべての集計計算を実行し、その後のCESPステートメントをそのにすることをおそらく最適です。
WITH cte
AS (SELECT bmreports_fpn_curves.bm_unit_name,
bmreports_fpn_curves.rundate,
bmreports_fpn_curves.period,
AVG(bmreports_fpn_curves.pn_level) AS fpn,
AVG(bmreports_fpn_curves.pn_level) AS boa,
MIN(bmreports_boa_curves.pn_level) minboa,
MAX(bmreports_fpn_curves.pn_level) maxfpn
FROM bmreports_fpn_curves
LEFT JOIN bmreports_boa_curves
ON bmreports_fpn_curves.bm_unit_name =
bmreports_boa_curves.bm_unit_name
AND bmreports_fpn_curves.rundate =
bmreports_boa_curves.rundate
GROUP BY bmreports_fpn_curves.bm_unit_name,
bmreports_fpn_curves.rundate,
bmreports_fpn_curves.period
HAVING bmreports_fpn_curves.bm_unit_name = 'T_DRAXX-1')
SELECT bm_unit_name,
rundate,
period ,
CASE
WHEN BOA IS NULL THEN FPN
WHEN BOA < FPN THEN MinBoa
WEHN BOA > FPN THEN MaxBoa
ELSE -- BOA = FPN THEN WHAT?
END as BOA
FROM cte
.
CTESをサポートしていないDBの場合は、FROM(インラインビュー)内部の選択を使用することもできます。なお、これはこれをサポートしています。
SELECT bm_unit_name,
rundate,
period ,
CASE
WHEN BOA IS NULL THEN FPN
WHEN BOA < FPN THEN MinBoa
WEHN BOA > FPN THEN MaxBoa
ELSE -- BOA = FPN THEN WHAT?
END as BOA
FROM (
SELECT bmreports_fpn_curves.bm_unit_name,
bmreports_fpn_curves.rundate,
bmreports_fpn_curves.period,
AVG(bmreports_fpn_curves.pn_level) AS fpn,
AVG(bmreports_fpn_curves.pn_level) AS boa,
MIN(bmreports_boa_curves.pn_level) minboa,
MAX(bmreports_fpn_curves.pn_level) maxfpn
FROM bmreports_fpn_curves
LEFT JOIN bmreports_boa_curves
ON bmreports_fpn_curves.bm_unit_name =
bmreports_boa_curves.bm_unit_name
AND bmreports_fpn_curves.rundate =
bmreports_boa_curves.rundate
GROUP BY bmreports_fpn_curves.bm_unit_name,
bmreports_fpn_curves.rundate,
bmreports_fpn_curves.period
HAVING bmreports_fpn_curves.bm_unit_name = 'T_DRAXX-1') ) t
. ケースにIIFのより文字通り翻訳を試みましたか?たとえば、IIFチェーンは次のようになります。
IIf
(
IIf
(
Min([dbo_BMReports_BOA_Curves]![PN_Level]) < [dbo_BMReports_FPN_Curves]![PN_Level],
Min([dbo_BMReports_BOA_Curves]![PN_Level]),
Max([dbo_BMReports_BOA_Curves]![PN_Level])
) Is Null,
[dbo_BMReports_FPN_Curves]![PN_Level],
IIf
(
Min([dbo_BMReports_BOA_Curves]![PN_Level]) < [dbo_BMReports_FPN_Curves]![PN_Level],
Min([dbo_BMReports_BOA_Curves]![PN_Level]),
Max([dbo_BMReports_BOA_Curves]![PN_Level])
)
) AS BOA
.
だからリテラル翻訳はこのようなものになるでしょう:
(
case
when
(
case
when Min(BMReports_BOA_Curves.PN_Level) < BMReports_FPN_Curves.PN_Level then
Min(BMReports_BOA_Curves.PN_Level)
else
Max(BMReports_BOA_Curves.PN_Level)
end
) is null then
BMReports_FPN_Curves.PN_Level
else
(
case
when Min(BMReports_BOA_Curves.PN_Level) < BMReports_FPN_Curves.PN_Level then
Min(BMReports_BOA_Curves.PN_Level)
else
Max(BMReports_BOA_Curves.PN_Level)
end
)
end
) as BOA
.
私はあなたの完全なスキーマやデータにアクセスできないので、翻訳をテストすることはできませんが、構文的に正しいと思います。