ホームページ >データベース >mysql チュートリアル >MySQL Advanced Learning: 効率的かつ適切なインデックスを作成する方法の詳細な説明
この記事は MySQL の高度な学習です。より適切なインデックスを作成する方法について詳しく説明します。お役に立てば幸いです。
ライブラリにデータが増えたときのインデックス作成の重要性がわかりません。ましてや、ライブラリにデータが増えたときの適切なインデックスもわかりません。インデックスの重要性。この記事では、効率的かつ適切なインデックスを作成する方法を紹介します。 [関連する推奨事項: mysql ビデオ チュートリアル ]
基本的にすべてのテーブルには主キーがあるため、通常の開発では、使用できる場合はインデックスを使用し、使用できる場合は主キー インデックスを使用します。
多くの場合、インデックスは実際には文字列であり、必然的に長い文字列が表示されるため、インデックスが多くのスペースを占有し、効率が低下します。特に、BLOB、テキスト、VARCHAR などの長い列の場合に適しています。このときの対処方法は、フィールドの全値をインデックスとして使用するのではなく、前半部分のみを取得することです(選択されたプレフィックスインデックスの選択性が列全体に近い)。これによりインデックス領域が大幅に削減され、効率が向上しますが、インデックスの選択性が低下するという欠点があります。
インデックス選択性: データ テーブル レコードの総数 (#T) に対する一意のインデックス値の比率 (1/#T ~ 1)。インデックスの選択性が高くなると、データが高度に区別され、より多くの行をフィルターで除外できるため、クエリ効率が高くなります。ユニークインデックスの選択性は 1 であり、そのパフォーマンスは最高です。
たとえば、会社の従業員テーブルの電子メール フィールドでは、会社の電子メール サフィックスは xxxx@qq.com
のように同じです。電子メールをインデックスとして使用する唯一の効果的な方法は、xxxx の部分です。@qq.com が同じであるため、インデックスとしては意味がありません。明らかに、xxxx のみがインデックスとして使用され、その選択性は値全体と同じです。ただし、xxxx をインデックスとして使用すると、明らかにインデックスのスペースが削減されます。
以下では例として従業員テーブルを使用します (テーブルの構造とデータについては記事の最後を参照)
電子メール フィールドを例としてインデックスを作成します。
このデータのメールアドレスは実際には携帯電話番号です@qq.com これを例に挙げると、実は最初の11桁とその後の桁はすべて同じです。次の SQL を使用して、これらのデータの選択性の計算を確認します (最初の 10、11、12 をそれぞれ取得します)。
-- 当是11个前缀的时候选择性是1,在增加字段长度,选择性也不会变化 select count(distinct left(email,10))/count(*) as e10, count(distinct left(email,11))/count(*) as e11, count(distinctleft(email,12))/count(*) as e12 from employee;上の図から、上位 10、上位 11、および上位 12 の選択性がそれぞれ 0.14、1.0、1.0 であり、インデックスの選択性が次のとおりであることがわかります。 11 番目の位置が最高の 1 なので、すべてをインデックスとして使用する必要がなく、インデックスのスペースが増加します。
-- 创建前缀索引 alter table employee add key(email(11));count を使用して統計の頻度を計算することもできます (出現回数が少ないほど、繰り返し率が低くなり、選択性が高くなります)
-- 查找前缀出现的频率 select count(*) as cnt,left(email,11) as pref from employee group by pref order by cnt desc limit 10;4. インデックス スキャンを使用して並べ替えます。並べ替えや並べ替えを使用する必要があることがよくありますが、並べ替えはパフォーマンスにさらに影響します。データ量が大きい場合は、データをメモリにロードして並べ替えます。メモリに保存され、複数のバッチでのみ処理できます。ただし、インデックス自体は順序付けされているため、インデックスを直接使用して並べ替えを完了する方が簡単です。 インデックスのスキャン自体は、1 つのインデックス レコードから次のレコードに移動するだけなので高速ですが、インデックスがクエリに必要なすべての列をカバーできない場合は、毎回インデックスをスキャンする必要があります。レコードがテーブルに返され、対応する行が 1 回クエリされます。これは基本的にランダム IO です。したがって、インデックス順でのデータの読み取りは、通常、連続した全テーブル スキャンよりも遅くなります。 Mysql は、ソートと行の検索に同じインデックスを使用できます。可能であれば、そのようなインデックスの作成を検討してください。
インデックス列の順序が order by 句の順序と完全に一致し、すべての列のソート方向 (後方または前方) が同じ場合にのみ、mysql はインデックスを使用できます。結果を並べ替えます。クエリを複数のテーブルに関連付ける必要がある場合、インデックス ソートは、order by 句のフィールドがすべて最初のテーブルのものである場合にのみ使用できます。クエリによる順序付けは、結合されたインデックスの左端のプレフィックスを満たす必要もあります。そうでない場合、インデックスの並べ替えは使用できません。
実は、開発中に注意すべき主な点が 2 つあります。上記のように、union all は次のようになります。そして、and または or は 1 回だけ実行されます。同時に、 or と in の実行計画は同じであることがわかります。\
ただし、それらの実行時間を調べています。以下に示すように、詳細な時間を表示するには set profiling=1
を使用し、特定の時間を表示するには show profiles
を使用します。次の図は、 or の時間が 0.00612000 で、 in の時間が 0.00022800 であることを示しています。ギャップは依然として非常に大きいです (テスト テーブル データには 200 行しかありません)
union all: クエリ 2 つの段階に分かれています。実際には、ユニオンもあります。日常の開発でユニオンを使用する必要がある場合は、ユニオンには個別の重複排除という追加のステップがあるため、ユニオン オールを使用することをお勧めします。したがって、union all を使用してみてください。
範囲条件を使用できます:>,>=,
範囲列インデックスは使用できますが、範囲列に続く列ではインデックスを使用できません (インデックスは最大 1 つの範囲列に使用できます)
たとえば、クエリ条件が where の場合、結合インデックス年齢名age>18 および name="Ji"
の後の名前はインデックス作成に使用されません。
面接で「ある指標に従うことができるということですか?」と聞かれたことがありますが、これまで気にしたこともうまく答えたこともありませんでした。今回は自分で実験してみました。結論としては記事の最後。
従業員テーブルに定義しましたmobile
フィールドは varchar 型でインデックスが付けられています。
limit 1 の制限を追加するために 1 つのデータだけが必要な場合、インデックス ポインタは、条件を満たすデータを見つけた後に停止します。はい、そのままリターンしてください。制限がない場合は判定が続行されます。
limit 10000,10005)、10,000 項目を走査して 5 つの項目を取得することになり、非常に非効率になります。ヒント: 主キーがシーケンシャルの場合は、主キーを介してページング データを直接取得できます。
インデックスは多ければ多いほど良いため、テーブルの構築を開始するときにインデックスを設計する必要はありません。時期尚早に最適化すると効率的なインデックスにはなりません。ビジネスを理解し、以下に基づいて行う必要があります。関連するインデックスを構築する前に、ビジネス SQL を統計的に評価する必要があります。これにより、考慮事項がより包括的になり、確立されたインデックスがより効果的かつ効率的になります。 上記はインデックスの最適化に対応する細かい詳細です。スムーズな SQL を書くのに役立つことを願っています。
補足
结论:只有主键会走,唯一键和普通索引都不会走。
我在employee表中建了唯一索引employee_num
和联合索引employee_num+name
,结果就是下图的执行情况。
employee表结构
CREATE TABLE `employee` ( `employee_id` bigint(20) NOT NULL AUTO_INCREMENT, `employee_num` varchar(30) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL COMMENT '员工编码', `name` varchar(60) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL COMMENT '员工姓名', `email` varchar(60) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NULL DEFAULT NULL COMMENT '电子邮件', `mobile` varchar(60) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NULL DEFAULT NULL COMMENT '移动电话', `gender` tinyint(1) NOT NULL COMMENT '性别, 0: 男 1: 女', PRIMARY KEY (`employee_id`) USING BTREE, INDEX `email`(`email`(11)) USING BTREE, INDEX `employee_u1`(`employee_num`, `name`) USING BTREE, UNIQUE INDEX `employee_u2`(`employee_num`) USING BTREE, INDEX `employee_u3`(`mobile`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 0 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_unicode_ci COMMENT = '员工表' ROW_FORMAT = Dynamic;
employee数据如下:
INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (10, '001', '员工A', '15500000001@qq.com', '15500000001', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (11, '002', '员工B', '15500000002@qq.com', '15500000002', 0); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (12, '003', '员工C', '15500000003@qq.com', '15500000003', 0); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (13, '004', '员工D', '15500000004@qq.com', '15500000004', 0); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (14, '005', '员工E', '15500000005@qq.com', '15500000005', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (15, '006', '员工F', '15500000006@qq.com', '15500000006', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (16, '007', '员工G', '15500000007@qq.com', '15500000007', 0); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (17, '008', '员工H', '15500000008@qq.com', '15500000008', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (18, '009', '员工I', '15500000009@qq.com', '15500000009', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (19, '010', '员工J', '15500000010@qq.com', '15500000010', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (20, '011', '员工K', '15500000011@qq.com', '15500000011', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (21, '012', '员工L', '15500000012@qq.com', '15500000012', 1); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (22, '013', '员工M', '15500000013@qq.com', '15500000013', 0); INSERT INTO `sakila`.`employee`(`employee_id`, `employee_num`, `name`, `email`, `mobile`, `gender`) VALUES (23, '014', '员工N', '15500000014@qq.com', '15500000014', 1);
更多编程相关知识,请访问:编程视频!!
以上がMySQL Advanced Learning: 効率的かつ適切なインデックスを作成する方法の詳細な説明の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。