Mysql関数呼び出しの最適化の詳細な説明

Mysql関数呼び出しの最適化の詳細な説明

関数呼び出しの最適化

MySQL 関数は、内部的に決定論的または非決定論的としてマークされます。引数に固定値を与えられた関数が、呼び出しごとに異なる結果を返す場合、その関数は未定義です。非決定論的関数の例: RAND()UUID()

関数が非決定的としてマークされている場合、 WHERE句内の関数への参照は、各行 (1 つのテーブルから選択する場合) または行の組み合わせ (複数のテーブル結合から選択する場合) に対して評価されます。

MySQL は、引数の型 (引数がテーブル列であるか定数値であるか) に基づいて、関数を評価するタイミングも決定します。テーブル列の値が変更されるたびに、テーブル列を引数として受け取る決定論的関数を評価する必要があります。

非決定論的関数はクエリのパフォーマンスに影響を与える可能性があります。たとえば、一部の最適化が利用できない場合や、より多くのロックが必要になる場合があります。以下の説明ではRAND()を使用しますが、他の不確実性関数にも適用できます。

テーブル t の定義が次のようになっているとします。

テーブル t を作成します (id INT NOT NULL PRIMARY KEY、col_a VARCHAR(100))。

次の 2 つのクエリを考えてみましょう。

tから*を選択します。WHERE id = POW(1,2);
SELECT * FROM t WHERE id = FLOOR(1 + RAND() * 49);

両方のクエリは、主キーとの等価比較により主キー検索を使用しているように見えますが、これは最初のクエリにのみ適用されます。

  • 定数引数を持つPOW()は定数値を取り、インデックス検索に使用されるため、最初のクエリでは常に最大 1 行が生成されます。
  • 2 番目のクエリには、非決定論的関数RAND()この関数はクエリ内の定数ではありませんが、実際にはテーブルの各行に新しい値 t があります。したがって、クエリはテーブルのすべての行を読み取り、各行の述語を評価し、主キーがランダム値と一致するすべての行を出力します。 id 列の値と RAND() シーケンスの値に応じて、0 行、1 行、またはそれ以上の行になります。

非決定性の影響はSELECTステートメントに限定されません。このUPDATEステートメントは、非決定論的関数を使用して変更する行を選択します。

t を更新します。set col_a = some_expr WHERE id = FLOOR(1 + RAND() * 49);

おそらく、主キーが式と一致する行を最大 1 つ更新することが意図されています。ただし、 id列の値とRAND()シーケンスの値に応じて、0 行、1 行、またはそれ以上の行が更新される場合があります。

今説明した動作は、パフォーマンスとレプリケーションに影響を及ぼします。

  • 非決定論的関数は定数値を生成しないため、オプティマイザーはインデックス シークなどの適用可能な他の戦略を使用できません。結果はテーブルスキャンになる可能性があります。
  • 一致する行に対して単一行ロックを取得する代わりに、 InnoDB範囲キー ロックにアップグレードする場合があります。
  • 実行された更新はレプリケーションにとって安全ではないと判断できませんでした。

困難さは、 RAND()テーブルの各行に対して関数を 1 回評価するという事実から生じます。複数関数の評価を回避するには、次のいずれかの手法を使用します。

  • 非決定的関数を含む式を別のステートメントに移動し、値を変数に格納します。元のステートメントで、式を、オプティマイザーが定数値として扱うことができる変数への参照に置き換えます。
@keyval = FLOOR(1 + RAND() * 49) を設定します。
UPDATE t SET col_a = some_expr WHERE id = @keyval;
  • 派生テーブル内の変数にランダムな値を割り当てます。この手法により、変数はWHERE句の比較に使用される前に値が割り当てられます。
optimizer_switch = 'derived_merge=off' を設定します。
t を更新します (SELECT @keyval := FLOOR(1 + RAND() * 49)) AS dt
col_a = some_expr を設定します。WHERE id = @keyval;

前述したように、 WHERE句内の非決定論的な式は最適化を妨げ、テーブル スキャンを引き起こす可能性があります。ただし、他の式が決定論的である場合、 WHERE句は部分的に最適化できます。例えば:

SELECT * FROM t WHERE partial_key=5 AND some_column=RAND();

オプティマイザーがpartial_keyを使用して選択された行のセットを削減できる場合、 RAND()実行回数が少なくなり、不確実性が最適化に与える影響を軽減できます。

上記はMySQL関数呼び出しの最適化の詳細な説明です。MySQL関数呼び出しの最適化の詳細については、123WORDPRESS.COMの他の関連記事に注目してください。

以下もご興味があるかもしれません:
  • MySQL 空間データストレージと関数
  • MySQL の日付型の単一行関数コードの詳細な説明
  • MySql の null 関数の使用の共有
  • MySQLの基本の共通機能
  • MySQL で sum 関数を使用する例のチュートリアル
  • MySQLの共通関数を使用してJSONを処理する方法
  • MySQLのDATE_ADDとADDDATE関数は、指定された時間間隔を日付に追加します。
  • MySQL関数の簡単な紹介

<<:  WeChatミニプログラムのすべてのページがログインされていることを確認する方法

>>:  CSS における zoom:1 属性の定義と機能

推薦する

MySQLのトランザクション管理操作の詳細な説明

この記事では、MySQL のトランザクション管理操作について説明します。ご参考までに、詳細は以下の通...

CSS ファイルをインポートする 4 つの方法 (インライン、インライン、外部、インポート) の詳細な説明

CSS インポート方法 - インラインスタイルタグ属性を通じて、CSSのキーと値のペアがタグに直接書...

react+antd.3x は IP 入力ボックスを実装します

この記事では、IP入力ボックスを実装するための react+antd.3x の具体的なコードを参考ま...

RR および RC 分離レベルでのインデックスとロックのテスト スクリプトのサンプル コード

基本概念現在の読み取りとスナップショットの読み取りMVCC では、読み取り操作はスナップショット読み...

一般的なDockerコマンドの概要

Dockerのインストール1. 要件: Linuxカーネルバージョン3.10以上 表示: uname...

Nginx プロキシ転送構成を通じてクロスドメイン API プロキシ転送を実装する方法

序文WEB 開発では、クロスドメイン リクエストが頻繁に発生します。クロスドメインの問題を解決する方...

CSS3 のディスプレイのグリッドレイアウトとフレックスレイアウトの詳細な説明

Gird レイアウトは Flex レイアウトといくつかの類似点があり、どちらもコンテナーの内部項目を...

VSCode の JS フォーマットでセミコロンを自動的に追加または削除する方法について

導入js コード文の末尾にセミコロンを追加しても追加しなくても問題ありません。一般的に、チームで開発...

PrometheusとGrafanaを使用したMySQLサーバーのパフォーマンス監視の詳細な説明

概要Prometheus は、HTTP プロトコルを介してリモート マシンからデータを収集し、ローカ...

MySQL 8.0.22 winx64 のインストールと設定のグラフィックチュートリアル

mysql 8.0.22 winx64のインストールと設定のグラフィックチュートリアルは参考までに、...

MySQL ビューの原理と使用法の詳細な分析

序文: MySQL では、ビューはおそらく最も一般的に使用されるデータベース オブジェクトの 1 つ...

JavaScript es6 における var、let、const の違いの詳細な説明

まず、よくある質問は、ECMAScript と JavaScript の関係は何ですか? ECMAS...

Eclipse/Tomcat でホットデプロイメントとホットスタートを実装する方法

1. ホット デプロイメント: コンテナの実行中にプロジェクト全体を再デプロイすることを意味します。...

ネイティブ JavaScript でシンプルな Gobang ゲームを実装する

この記事では、JavaScriptで簡単なGobangゲームを実装するための具体的なコードを参考まで...

Win7 x64 に解凍版の mysql 5.7.18 winx64 をインストールするとサービスが起動できない問題を解決します

今日、mysql の公式サイトから mysql-5.7.18-winx64.zip をダウンロードし...