1. ホーム
  2. sql

[解決済み] WHERE句の参照エイリアス(SELECTで計算)。

2022-04-27 07:59:20

質問

SELECT (InvoiceTotal - PaymentTotal - CreditTotal) AS BalanceDue
FROM Invoices
WHERE BalanceDue > 0 --error

計算された値 BalanceDue 選択されたカラムのリストで変数として設定されているものは、そのカラムに対応する WHERE 節があります。

できる方法はありますか?この関連する質問で ( MySQLのSelectステートメントでWhere句に変数を使用する )、答えは、実は、ダメで、計算を書き出すだけのような気がします( は、クエリ内でその計算を行う)を2回行い、いずれも満足のいくものではありません。

どうすればいいですか?

SELECTは2番目に評価される最後の節なので、ORDER BY以外ではエイリアスを参照できません。2つの回避策があります。

SELECT BalanceDue FROM (
  SELECT (InvoiceTotal - PaymentTotal - CreditTotal) AS BalanceDue
  FROM Invoices
) AS x
WHERE BalanceDue > 0;

または、式を繰り返すだけです。

SELECT (InvoiceTotal - PaymentTotal - CreditTotal) AS BalanceDue
FROM Invoices
WHERE  (InvoiceTotal - PaymentTotal - CreditTotal)  > 0;

私は後者がいいと思います。式が非常に複雑な場合(あるいは計算にコストがかかる場合)、代わりに計算カラム(およびおそらくパーシステッド)を考慮する必要があります。

追記 あなたの心配は杞憂に終わったようです。少なくともこの単純な例では、SQL Serverは賢いので、たとえ2回参照したとしても、計算は1回しか実行しません。計画を比較してみてください。もし、もっと複雑なケースで式が複数回評価されるようなことがあれば、より複雑なクエリと計画を投稿してください。

以下は5つのクエリの例で、すべて全く同じ実行プランになります。

SELECT LEN(name) + column_id AS x
FROM sys.all_columns
WHERE LEN(name) + column_id > 30;

SELECT x FROM (
SELECT LEN(name) + column_id AS x
FROM sys.all_columns
) AS x
WHERE x > 30;

SELECT LEN(name) + column_id AS x
FROM sys.all_columns
WHERE column_id + LEN(name) > 30;

SELECT name, column_id, x FROM (
SELECT name, column_id, LEN(name) + column_id AS x
FROM sys.all_columns
) AS x
WHERE x > 30;

SELECT name, column_id, x FROM (
SELECT name, column_id, LEN(name) + column_id AS x
FROM sys.all_columns
) AS x
WHERE LEN(name) + column_id > 30;

5つのクエリすべてに対する結果プラン。