この 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アプレットで計算機機能を実装する

推薦する

Vueのコンポーネント値の転送から始まるオブザーバーモードの詳細な説明

目次オブザーバーパターンVue パス値最初のステップは、main.jsにバスを登録することです。 2...

MySQL のデータ型とスキーマの最適化の詳細な説明

現在、MySQL の最適化について学習しています。この記事では、データ型とスキーマの最適化について紹...

Vueカウンターの実装

目次1. カウンターの実装2. 成果を達成する1. カウンターの実装ページにカウンターを実装するだけ...

CSS の border 属性と display 属性の使い方の簡単な分析

境界プロパティの概要borderプロパティは要素の境界を設定します。境界線の3要素は、太さ、線の種類...

jsで七夕告白連打の効果を実現、jQueryで連打技術を実現

この記事では、jsとjQueryテクノロジーを使用して告白弾幕を実現する方法を紹介します。具体的な内...

docker コンペ応募でよく使われるコマンドのまとめ

アカウントにログイン DOCKER_REGISTRY=registry.cn-hangzhou.al...

MySQL スロークエリログの設定と使用方法のチュートリアル

序文MySQL スロー クエリ ログは、日常業務でよく遭遇する機能です。MySQL スロー クエリ ...

子コンポーネントで vue activated を使用する詳細

ページ: ベース: <テンプレート> <div class="タブコンテ...

MySQLデータベースに画像を保存するいくつかの方法

通常、ユーザーがアップロードした写真はデータベースに保存する必要があります。一般的に、解決策は 2 ...

jwtを使用してノードによって生成されたトークンをどこに保存するかについての簡単な説明

A: 通常はクライアントに保存されます。 jwt または JSON Web Token は、リクエス...

React で Antd の Form コンポーネントを使用してフォーム機能を実装する方法

1. 構造部品1. フォームには、入力コントロール、標準フォーム フィールド、ラベル、ドロップダウン...

MySQL コマンドを使用してインデックスを作成、削除、およびクエリする方法の紹介

MySQL データベース テーブルでは、インデックスを作成、表示、再構築、削除できるため、クエリ速度...

ES6拡張演算子の理解と使用シナリオ

目次1. 適用メソッドを置き換え、関数を呼び出すときにパラメータを処理する2. 残りパラメータ(残り...

Vueスロットの詳細な説明

1. 機能: 親コンポーネントが子コンポーネントの指定された位置に HTML 構造を挿入できるように...

HTML テーブルタグチュートリアル (23): 行の境界線の色属性 BORDERCOLORDARK

行ごとに、暗い境界線の色を個別に定義できます。基本的な構文<TR 境界線の色を暗くする=col...