(4)partitions
クエリはレコードのパーティションと一致します。値 NULL
は、パーティション化されていないテーブルに使用されます。
(5)type
最良から最悪まで:
system
>const
>eq_ref
>ref
>ref_or_null
>range
>index
> ALL
all
に加えて、index_merge
を除く他のすべての type
はインデックスを使用できます。他の type
はインデックスを 1 つだけ使用できます。
一連のテーブルを自分で作成して実験してみましょう:
SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;
-- ----------------------------
-- Table structure for goods
-- ----------------------------
DROP TABLE IF EXISTS `goods`;
CREATE TABLE `goods` (
`id` int(11) NOT NULL,
`sn` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NULL DEFAULT NULL,
`name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NULL DEFAULT NULL,
PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_0900_ai_ci ROW_FORMAT = Dynamic;
-- ----------------------------
-- Records of goods
-- ----------------------------
INSERT INTO `goods` VALUES (1, 'sn123456', '衣服');
-- ----------------------------
-- Table structure for sku
-- ----------------------------
DROP TABLE IF EXISTS `sku`;
CREATE TABLE `sku` (
`id` int(11) NOT NULL,
`goods_id` int(11) NOT NULL,
`status` int(11) NOT NULL,
`deleted` int(11) NOT NULL,
`barcode` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NULL DEFAULT NULL,
`name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NULL DEFAULT NULL,
PRIMARY KEY (`id`) USING BTREE,
UNIQUE INDEX `index_2`(`name`) USING BTREE,
INDEX `index_1`(`goods_id`, `status`, `deleted`, `barcode`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_0900_ai_ci ROW_FORMAT = Dynamic;
-- ----------------------------
-- Records of sku
-- ----------------------------
INSERT INTO `sku` VALUES (1, 1, 1, 0, 'kt123456', '黑色');
SET FOREIGN_KEY_CHECKS = 1;
ログイン後にコピー
system
テーブルにはレコードが 1 行だけあります (システム テーブルと同じです)。これは const 型の特殊なケースです。通常は表示されないため、無視できます。
const
# はインデックスを通じて 1 回見つかったことを意味し、const は主キーまたは一意のインデックスを比較するために使用されます。レコードの 1 行のみが照合されるため、非常に高速です。 where リストに主キーを配置すると、mysql はクエリを定数に変換できます;
EXPLAIN SELECT * FROM sku WHERE id=1;复制代码
ログイン後にコピー
eq_ref
unique性的インデックス スキャン。インデックス キーごとにテーブル内の 1 つのレコードのみがそれに一致し、主キーまたは一意のインデックス スキャンによく使用されます。このタイプは通常、複数のテーブルの結合同等クエリに表示されます。つまり、前のテーブルの各結果に対して、 , 後者のテーブルの結果は 1 行のみ照合できるため、クエリ効率が高くなります。
EXPLAIN SELECT * FROM sku,goods WHERE sku.goods_id=goods.id;
ログイン後にコピー
ref
非一意のインデックス スキャンは、単一の値に一致するすべての行を返します。これは本質的にインデックス アクセスです。単一の値に一致するすべての行を返しますが、複数の一致する行が見つかる可能性があるため、検索とスキャンのハイブリッドである必要があります;
EXPLAIN SELECT * FROM sku WHERE goods_id=1;
ログイン後にコピー
ref_or_null
セカンダリ インデックスの等しい値の比較も null に制限されます。
EXPLAIN SELECT * FROM sku WHERE name='123456' or name IS NULL;
ログイン後にコピー
range
インデックスを使用して行を選択し、指定された範囲内の行のみを取得します。キー列はどの行を示すかを示します。インデックスが使用され、通常、 between、<、>、in などのクエリが where ステートメントに表示されます。この範囲インデックス スキャンは、テーブル内の特定の時点で開始するだけでよいため、フル テーブル スキャンよりも優れています。 1 つのポイント、すべてのインデックスをスキャンする必要はありません;
EXPLAIN SELECT * FROM sku WHERE id BETWEEN 1 and 10;
ログイン後にコピー
index
インデックスの違いそしてすべては、インデックス タイプがインデックス ツリーのみをトラバースするということです。これは、インデックス ファイルが通常データ ファイルより小さいため、通常は all より高速です。つまり、all とインデックスはテーブルの読み取りと書き込みの両方を行いますが、インデックスはインデックスから読み取ります。ハードディスクからのすべての読み取り;
EXPLAIN SELECT barcode FROM sku WHERE deleted=0;
ログイン後にコピー
##all# はフルテーブルスキャン;
EXPLAIN SELECT * FROM sku WHERE deleted=0;
ログイン後にコピー
(6) possible_keys
クエリで使用できるインデックスがここにリストされます。
(7) key
は実際に使用されるインデックスをクエリします。
select_type が index_merge
の場合、2 つのエラーが発生する可能性があります。複数のインデックスの場合、他の select_type
のうちの 1 つだけがここに表示されます。
(8)key_len
key_len は、列計算クエリで使用されるインデックスの長さを示します。例:
SELECT * FROM table where age = 1 and name like 'xx' (age は int 型で null は不可、name は varchar(20) 型で null 可) と仮定します。エンコードはutf8です。これら 2 つのフィールドがインデックス クエリとして使用される場合、key_len の値は 4 3 * 20 2 1 = 67
になります。具体的な計算ルールを次の表に示します。
#値の種類 | 値の名前 | 説明 |
文字列 | CHAR(n) | n バイト長 |
##VARCHAR(n) |
の場合utf8 エンコードの場合は 3 n 2 バイト、utf8mb4 エンコードの場合は 4 n 2 バイトです。 |
| #数値型
TINYINT | 1バイト |
|
SMALLINT
| 2 バイト |
|
MEDIUMINT
| 3 バイト |
|
INT
| 4 バイト |
|
##BIGINT
8 バイト |
|
時間タイプ |
DATE
3バイト |
|
|
TIMESTAMP
4 バイト |
|
|
DATETIME
#8 バイト
|
#フィールド属性 |
NULL 属性は 1 バイトを占めます。フィールドが NOT NULL の場合、そのフィールドは占有されていません。 |
|
|
(9)ref
定数等価クエリの場合はここにconst
が表示され、接続クエリの場合は実行計画が表示されます。駆動テーブル ドライバ テーブルの関連フィールドがここに表示されます。条件で式または関数が使用されている場合、または条件列が内部で暗黙的な変換を受ける場合、func
と表示されることがあります。
(10)rows
これは、実行計画内のスキャン行の推定数であり、正確な値ではありません。
(11)filtered
この列は、explain extend
を使用するときに表示されます。5.7# 以降のバージョンではデフォルトで利用可能です。 ## このフィールドには
explain extend を使用する必要はありません。このフィールドは、ストレージ エンジンから返されたデータがサーバー層でフィルタリングされた後、クエリを満たす残りのレコードの割合を示します。これは特定のレコード数ではなくパーセンテージであることに注意してください。
(12)追加
この列には多くの情報が表示されます。情報は数十あります。一般的に使用されるものは次のとおりです。
1、distinct: distinct キーワードは
select 部分で使用されます
2、テーブルは使用されません: # なし##from クエリまたは From デュアル
クエリ。 not in()
フォーム サブクエリまたは notexists()
演算子を使用してクエリを結合します。これはアンチ結合と呼ばれます。つまり、一般的な結合クエリは最初に内部テーブルをクエリし、次に外部テーブルをクエリしますが、アンチ結合クエリは最初に外部テーブルをクエリし、次に内部テーブルをクエリします。
3. filesort の使用: 説明 mysql は、テーブル内のインデックスの順序でデータを読み取るのではなく、外部インデックスを使用してデータを並べ替えます。 mysql
のインデックスを使用して完了できない並べ替え操作は、「ファイルの並べ替え」と呼ばれます。これは、ソート中にインデックスが使用できない場合に発生します。 order by
ステートメントでよく見られるのですが、できるだけ早く最適化する必要があります
4.インデックスを使用する: テーブルに戻る必要はありません。クエリを実行し、クエリ データはインデックスを介して直接取得できます。
5. 結合バッファー (ブロック ネスト ループ) の使用、結合バッファー (バッチ化されたキー アクセス) の使用 : 5.6.x 以降のバージョンは関連クエリを最適化します BNL
、BKA
の機能。主な目的は、内部テーブルのループ数を減らし、クエリを順番にスキャンすることです。
6. sort_union の使用、union の使用、intersect の使用、sort_intersection の使用: ##intersect の使用:
と の各インデックスの使用を示します。条件を使用した場合、処理結果から共通部分を求めることを示します -
using Union:
or を使用してインデックスを使用して条件を結合することを示します-
using sort_union と using sort_intersection: 前の 2 つと似ていますが、
and と - or
を使用して実行する場合に表示される点が異なります。大量の情報をクエリするには、まず主キーをクエリしてから続行します。レコードを読み取って返すことができるのは、並べ替えとマージを行った後でのみです。
7. usingtemporary
: 中間結果を保存するために一時テーブルが使用されることを示します。一時テーブルはメモリ一時テーブルまたはディスク一時テーブルです。実行プランでは確認できません。status 変数、used_tmp_table、
used_tmp_disk_table を確認する必要があります。 ### それを見るために。 order by
およびグループ クエリ group by
でよく見られます。 group by
作成されたインデックスの順序と数に従ってください。できるだけ早く最適化する必要があります
8. where
を使用: ストレージ エンジンから返されたすべてのレコードがクエリ条件を満たしているわけではないため、
server# でフィルターする必要があることを示します。 ## レイヤー。クエリ条件は制限条件と検査条件に分かれています。5.6 より前は、ストレージ エンジンはデータをスキャンして制限条件に従って返すことしかできず、その後、server レイヤーがそれに応じてフィルタリングしていました。検査条件に合わせて実際に条件を満たしたデータを返す データを問い合わせます。
5.6.x 以降、
ICP 機能 (インデックス条件プッシュダウン、インデックス プッシュダウン) がサポートされ、チェック条件をストレージ エンジン層までプッシュダウンすることもできます。チェック条件と制限は直接読み取りません。これにより、ストレージ エンジンによってスキャンされるレコードの数が大幅に減少します。
extra列表示
インデックス条件を使用
9, firstmatch(tb_name): 5.6.x
から導入されたオプティマイザ 1 つクエリの新機能は、in() タイプのサブクエリを含む where 句で共通です。内部テーブルのデータ量が比較的大きい場合、この問題が発生する可能性があります。
10、loosescan(m..n):
5.6.x## の後に導入された最適化サブクエリ# 新しい機能の 1 つは、
in() 型のサブクエリで、サブクエリが重複レコードを返すときに、この問題が発生する可能性があることです。4. 主な懸念事項の説明
一般に、結果のいくつかの列に注目するだけで済みます:
#列名 | 備考 |
type | このクエリ テーブル 接続タイプここから、このクエリのおおよその効率を確認できます。 |
key | 最終的に選択されたインデックス。インデックスがない場合、通常、このクエリの効率は非常に低くなります。 |
#key_len
このクエリの結果フィルタリングに使用されるインデックスの実際の長さ |
|
rows
estimate スキャンする必要があるレコードの数、スキャンする必要があるレコードの推定数 | 小さいほど良い |
Extra
Extra | #追加の追加情報。主に表示されるかどうかを確認するためですfilesort を使用する 、 一時ファイルを使用する これら 2 つの場合
|
##注意が必要な Extra
列を見てみましょう。いくつかの状況:
キーワード
備考 |
|
#ファイルソートの使用
結果をインデックス順に並べるのではなく、外部の並べ替えを使用します。データが小さい場合は、ファイルソートの順序から並べ替えられます。それ以外の場合は、ディスク上で並べ替えを完了する必要があり、非常にコストがかかります。 適切なインデックスを追加する必要があります |
一時的なものを使用する |
結果を保存するために一時テーブルを作成する必要があります。これは通常、インデックスのない列に対して GROUP BY を実行する場合、または | ORDER BY のすべての列がインデックスに含まれていない場合に発生します。 適切なインデックスを追加する必要があります インデックスの使用 |
意味MySQL テーブル全体のスキャンを避けるためにカバーインデックスを使用しますまた、テーブル内のデータを 2 回検索する必要がなく、これはより良い結果の 1 つです。 | type
Using where is a full table/ の | index
タイプと混同しないように注意してください。完全なインデックス スキャン後、WHERE 句を使用して結果のフィルタリングを完了します。 | 適切なインデックスを追加する必要があります
不可能 WHERE |
YesWhere句の判定結果は常に false となり、データは選択できません (例: | where 1=0)。あまり注意する必要はありません
最適化されたテーブルの選択 |
#一部の集計関数を使用してインデックスのあるフィールドにアクセスする場合、オプティマイザはインデックスを介して必要なデータ行を直接見つけて、全体の処理を完了します。 MIN()\MAX () などのクエリ、これもより良い結果の 1 つです |
| [関連する推奨事項:
mysql ビデオチュートリアル###]###