この SQL 書き込み方法では本当にインデックスが失敗するのでしょうか?

この SQL 書き込み方法では本当にインデックスが失敗するのでしょうか?

序文

インターネット上には、MySQL でインデックスにヒットできないさまざまな状況をまとめた記事がよくあります。その 1 つは、またはを使用するステートメントがインデックスにヒットできないというものです。

この記述は実際には正しくありません。正しい結論は、MySQL 5.0 以降では、またはで接続されたフィールドに独立したインデックスがある場合、そのインデックスにヒットできるということです。ここで、index_merge 機能が使用されます。

MySQL 5.0 より前では、SQL ステートメントは 1 つのインデックスしか使用できません。SQL ステートメントで or キーワードを使用すると、既存のインデックスは無効になり、テーブル全体のスキャンが実行されます。どのインデックスを使用しても、MySQL は条件を満たすデータを一度に見つけることができないため、インデックスを放棄することしかできません。

MySQL も継続的にアップグレードおよび更新されており、MySQL バージョン 5.0 以降では index_merge インデックス マージ機能が追加され、1 つの SQL で複数のインデックスを使用することも可能になりました。

index_merge の基本的な考え方は、まず単一のインデックスを使用して要件を満たすデータを見つけ、次にそれらのデータを結合して返すことです。
例を見てみましょう。

ここでも前回の記事で作成したテーブルとテストデータを使用します。10w のテストデータがテーブルに挿入されます。テーブル構造は次のとおりです。

テーブル `t` を作成します (
 `id` int(11) NULLではない、
 `a` int(11) デフォルト NULL,
 `b` int(11) デフォルト NULL,
 主キー (`id`)
)ENGINE=InnoDB;

まず a フィールドにインデックスを追加し、次に または を使用してクエリ ステートメントを実行して、それがどのように機能するかを確認しましょう。

mysql> テーブル t を変更し、インデックス a_index(a) を追加します。
クエリは正常、影響を受けた行は 0 行 (0.17 秒)
レコード: 0 重複: 0 警告: 0
mysql> explain select a from t where a=100 or b=6000;
+----+-------------+-------+-------+---------------+-------+--------+---------+------------+------------+
| id | select_type | テーブル | タイプ | possible_keys | key | key_len | ref | 行 | 追加 |
+----+-------------+-------+-------+---------------+-------+--------+---------+------------+------------+
| 1 | SIMPLE | t | ALL | a_index | NULL | NULL | NULL | 100332 | where の使用 |
+----+-------------+-------+-------+---------------+-------+--------+---------+------------+------------+
セット内の 1 行 (0.00 秒)

フィールド b にはインデックスがないため、MySQL はテーブルを返すプロセスを回避できるため、完全なテーブル スキャンの方がコストがかからないと判断します。

次に、フィールド b にインデックスを追加して、SQL ステートメントを再度実行します。

mysql> テーブル t を変更し、インデックス b_index(b) を追加します。
クエリは正常、影響を受けた行は 0 行 (0.17 秒)
レコード: 0 重複: 0 警告: 0
mysql> explain select a from t where a=100 or b=6000;
+----+-------------+-------------+------------+-----------------+--------+------+------+-------------------------------------------+
| id | select_type | テーブル | タイプ | possible_keys | key | key_len | ref | 行 | 追加 |
+----+-------------+-------------+------------+-----------------+--------+------+------+-------------------------------------------+
| 1 | SIMPLE | t | index_merge | a_index,b_index | a_index,b_index | 5,5 | NULL | 2 | union(a_index,b_index); を使用する where を使用する |
+----+-------------+-------------+------------+-----------------+--------+------+------+-------------------------------------------+
セット内の 1 行 (0.00 秒)

今回は、MySQL がインデックス a と b の両方を使用し、type フィールドの値が index_merge であることがわかります。

次に、別の SQL ステートメントを見て、結果がどうなるかを確認してみましょう。

mysql> explain select a from t where a>100 or b>6000;
+----+-------------+-------+-------+-----------------+-------+--------+--------+-----------+------------+
| id | select_type | テーブル | タイプ | possible_keys | key | key_len | ref | 行 | 追加 |
+----+-------------+-------+-------+-----------------+-------+--------+--------+-----------+------------+
| 1 | SIMPLE | t | ALL | a_index,b_index | NULL | NULL | NULL | 100332 | where の使用 |
+----+-------------+-------+-------+-----------------+-------+--------+--------+-----------+------------+
セット内の 1 行 (0.00 秒)

この SQL 文は、等号を大なり記号に変更するだけなので、返される結果セットは間隔セットになります。MySQL はここでインデックスを放棄し、テーブル全体のスキャンを実行します。ただし、いくつかの記事で、この問題は MySQL バージョン 5.7 以降で最適化されている、つまり、間隔クエリでも index_merge がサポートされていると読みました。私のバージョンは 5.6 で、この最適化はまだ検証していません。興味があれば、検証してみてください。

実際、MySQL には絶対的なことがたくさんあります。MySQL のバージョンが異なれば、同じ SQL に対しても内部処理方法が異なる場合があります。同時に、MySQL は継続的に最適化およびアップグレードされており、古い知識ポイントの一部は簡単に適用できなくなることがわかります。

この記事が皆さんのお役に立てば幸いです。フォローして「いいね!」していただければ、私にとって最高のサポートになります。ありがとうございます。

また、MySQL の基盤となるデータ構造については、私が以前に書いた他の記事を参照すると、この記事の理解に役立つかもしれません。

要約する

以上がこの記事の全内容です。この記事の内容が皆様の勉強や仕事に何らかの参考学習価値をもたらすことを願います。123WORDPRESS.COM をご愛顧いただき、誠にありがとうございます。

以下もご興味があるかもしれません:
  • Mysql インデックスが失敗するいくつかの状況の分析
  • MySQL インデックス障害の 5 つの状況の分析
  • MySQL でデータベース インデックスが失敗する状況の詳細な分析
  • mysql はインデックスを無効にしますか?

<<:  CentOS7 は Docker のバージョン 19 をデプロイします (簡単なので、従ってください)

>>:  WeChatアプレットで計算機機能を実装する

推薦する

プロジェクトにおける CSS グリッドシステムの柔軟な使用方法の詳細な説明

序文CSS グリッドは通常、さまざまなフレームワークにバンドルされていますが、実際のビジネス ニーズ...

Linuxでディスク使用量を確認する方法

1. dfコマンドを使用してディスク全体の使用量を表示します。 df コマンドは、ハードディスクのマ...

仮想マシン VMware に Kali Linux をインストールする最新の超詳細なグラフィック チュートリアル

目次1. システムイメージファイルをダウンロードする2. 新しい仮想マシンを作成する3. Kali ...

JQueryはアニメーション効果の非表示と表示を実装します

この記事では、アニメーション効果の非表示と表示を実現するためのJQueryの具体的なコードを参考まで...

プライベートレジストリ内の画像を照会または取得する方法

Dockerはプライベートレジストリ内のイメージを照会または取得するために、 docker 検索 1...

JavaScript インスタンス オブジェクトでプロトタイプ メソッドをオーバーライドする方法の詳細

目次JavaScriptでは、通常、次のコードのようにクラスを簡単に定義できます。 var サンプル...

MySQL ログトリガー実装コード

SQL文 ドロップトリガー もし sys_menu_edit が存在します。 各行のsys_menu...

MLSQLコンパイル時権限制御例の詳細な説明

序文MySQL の権限を簡単に理解すると、MySQL では自分の能力の範囲内で操作が許可され、その限...

MySQLからClickHouseに移行する5つの方法

データ移行は、MySQL から ClickHouse にインポートする必要があります。概要プランは以...

MySQLデータベーステーブルの容量を確認する方法の例

この記事では、MySQL のデータベース テーブルの容量を確認するためのコマンド ステートメントを紹...

Vue で SVG アイコンを導入する 2 つの方法

Vue で SVG アイコンを導入する方法Vue で svg アイコンを導入する方法 1インストール...

Vue で Openlayer を使用して読み込みアニメーション効果を実現する

注意: スコープアニメーションは使用できません。 ! ! ! GIF経由 <テンプレート>...

Vue モバイル開発で better-scroll を使用するときにクリック イベントが失敗する問題の解決策

最近、モバイル プロジェクトの開発方法を学ぶために vue を使用し、スクロールには better-...

Docker コンテナ ソース コードのデプロイ httpd ストレージ ボリュームを使用して Web サイトをデプロイする (推奨)

目次Dockerコンテナのソースコードを使用してhttpdをデプロイし、ストレージボリュームを使用し...

エラー 2002 (HY000): ソケット '/tmp/mysql.sock' 経由でローカル MySQL サーバーに接続できません

エラーメッセージ:エラー 2002 (HY000): ソケット '/tmp/mysql.so...