質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.77%

CASE式を書くと実行計画では早くなったが実際の処理速度は遅くなった

解決済

回答 3

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 2,489

key_FoolyCooly

score 16

前提・実現したいこと

あるクエリの性能改善を行ったさい、
実行計画ではIndex Scan → Index Scan + Key lookup程度には推定コストと推定行数が下がっているのに、
画面操作によりクエリをプロファイラで確認するとTable Scanかというくらい両者とも上がることがありました。

実行計画では早くなるのに、実際の画面操作だと遅くなる原因を知りたいです。

質問のタイトルには「CASE式~」とありますが、CASE式が直接の原因かは不明です。
また、私はこういう場合にCASE式を使うと早くなると考えているのですが、間違っていたら教えてください。。

該当のソースコードが曖昧なため、
「もしかして」レベルでも良いので心当たりあれば教えてくださると幸いです。

該当のソースコード

SELECT
    ...
    FROM
        FUGA
    WHERE
        (@hoge0 == ''
            OR @hoge1 is Null
            OR FUGA.name = @hoge2)
        AND
        (@foo0== ''
            OR @foo1 is Null
            OR FUGA.age = @foo2)
        AND
            ...


SELECT
    ...
    FROM
        FUGA
    WHERE
        1 = 
            CASE
                WHEN FUGA.name = @hoge2
                    THEN 1
                WHEN @hoge0 == ''
                    THEN 1
                WHEN @hoge1 is Null
                    THEN 1
                ELSE
                    0
            END
        AND
        1 = 
            CASE
                WHEN FUGA.name = @foo2
                    THEN 1
                WHEN @foo0== ''
                    THEN 1
                WHEN @foo1 is Null
                    THEN 1
                ELSE
                    0
            END
        AND
            ...

補足情報(FW/ツールのバージョンなど)

  • SQL Server
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 3

check解決した方法

0

クエリの生成を条件分岐することで
問題を解決できました。
質問が解決したわけではないですが、
回答ありがとうございます。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

0

統計情報の更新状況の確認 統計情報が古いとか?
質問にCREATE TABLE, CREATE INDEX, 最初と修正後の実行計画も載せては?

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

0

caseで項目の比較について分岐させるようにする場合、少なくとも式の片側がindexの項目に一致しているようにして、オプティマイザがindexを使用しやすくするように工夫した方が良いですね。

select *
from FUGA
where FUGA.name=case when @hoge0 == '' or @hoge0 is Null then FUGA.name else @hoge2 end
  and FUGA.name=case when @foo0== '' or @foo0 is Null then FUGA.name else @foo2 end

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.77%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る