【MySQL】深层理解索引及特性(重点)--下(12)

索引(重点)

  • 1. 索引的作用
  • 2. 索引操作
    • 2.1 主键索引
      • 2.1.1 主键索引的特点
      • 2.1.2 创建主键索引
    • 2.2 唯一键索引
      • 2.2.1 唯一键索引的特点
      • 2.2.2 唯一索引的创建
    • 2.3 普通索引
      • 2.3.1 普通索引的特点
      • 2.3.2 普通索引的创建
    • 2.4 全文索引
      • 2.4.1 全文索引的作用
      • 2.4.2 全文索引的创建
  • 3. 查询索引
  • 4. 删除索引
  • 5. 索引创建原则

1. 索引的作用

索引提高数据库的性能,索引是物美价廉的东西了。不用加内存,不用改程序,不用调sql,只要执行正确的create index,查询速度就可能提高成百上千倍。但是天下没有免费的午餐,查询速度的提高是以插入、更新、删除的速度为代价的,这些操作,增加了大量的IO。所以它的价值,在于提高一个海量数据的检索速度。

常见索引类型分为:

  1. 主键索引(Primary Key)
    主键索引是一种特殊的唯一索引,不允许有重复值,并且每个表只能有一个主键。主键通常用于唯一标识表中的每一行数据。在创建表的时候,如果指定了某列为主键,那么该列会自动创建一个主键索引。
  2. 唯一索引(Unique)
    唯一索引确保了索引列中的所有值都是唯一的,但与主键不同的是,一个表可以有多个唯一索引。唯一索引允许有一个或多个NULL值存在,这取决于数据库系统的设计。
  3. 普通索引(Index)
    普通索引是最基本的索引类型,它没有任何限制,可以包含重复的值。通过在查询条件中频繁使用的列上创建普通索引,可以显著提高查询效率。
  4. 全文索引(Fulltext
    全文索引主要用于全文本搜索,它可以对文本内容进行复杂的搜索操作,比如查找包含特定单词或短语的记录。全文索引特别适合于处理大量的文本数据,如新闻文章、博客帖子等。值得注意的是,不同的数据库管理系统支持的全文索引功能可能有所不同,例如MySQL中的InnoDB和`MyISAM存储引擎都支持全文索引,但实现方式和性能特点可能有所区别。

示例:
当我们在数据量少的表中查询数据不会发现主键查询和普通查询之间的差异。

创建一个海量表:

--构建一个8000000条记录的数据--构建的海量表数据需要有差异性,所以使用存储过程来创建, 拷贝下面代码就可以了,暂时不用理解-- 产生随机字符串
delimiter $$create function rand_string(n INT)returns varchar(255)begin 
declare chars_str varchar(100) default'abcdefghijklmnopqrstuvwxyzABCDEFJHIJKLMNOPQRSTUVWXYZ';declare return_str varchar(255) default '';declare i int default 0;while i < n do 
set return_str =concat(return_str,substring(chars_str,floor(1+rand()*52),1));set i = i + 1;end while;return return_str;end $$delimiter ;--产生随机数字
delimiter $$create function rand_num()returns int(5)begin 
declare i int default 0;set i = floor(10+rand()*500);return i;end $$delimiter ;--创建存储过程,向雇员表添加海量数据
delimiter $$create procedure insert_emp(in start int(10),in max_num int(10))begindeclare i int default 0; 
set autocommit = 0;  
repeatset i = i + 1;insert into EMP values ((start+i) 
,rand_string(6),'SALESMAN',0001,curdate(),2000,400,rand_num());until i = max_numend repeat;commit;end $$delimiter ;-- 执行存储过程,添加8000000条记录
call insert_emp(100001, 8000000);

注:已经绑定资源,可以自行下载。

进入到一个数据库,用soure命令导入数据:
我的数据包index_data.sql /root/MySQL_data/some 目录下
命令:

//进入到test数据库
use test// 导入数据
source /root/MySQL_data/some/index-data.sql

这是一个有8000000条记录的数据,需要等一会儿,我花费了7分钟。
在这里插入图片描述

此时的表还没有创建索引,进行查询
命令:select * from EMP where empno = 188888
在这里插入图片描述
查询花费了4.69秒,这是在本机一个人来操作,在实际项目中,如果放在公网,假如同时有
1000个人并发查询,那很可能就死机

怎么让查询变得快呢?
答:创建索引

创建索引
命令:alter table EMP add index(empno);
在这里插入图片描述
创建索引花费了26.7秒。

再次查询
命令:select * from EMP where empno = 188888;
在这里插入图片描述
当执行查询的时候会明显的变快。

2. 索引操作

2.1 主键索引

2.1.1 主键索引的特点

  1. 唯一性:主键索引确保了索引列中的每一个值都是唯一的,不允许出现重复值。这是主键索引的核心特性,确保了每一行数据在表中都有一个唯一的标识符。
  2. 非空性:主键索引的列不允许有NULL值。这意味着在插入或更新数据时,必须为该列提供一个有效的、非空的值。
  3. 自动创建:当您在创建表时指定某个列为PRIMARY KEY时,数据库会自动为该列创建一个主键索引。如果表中没有显式定义主键,某些数据库系统可能会自动创建一个隐式的主键(例如,SQL Server中的IDENTITY列)。
  4. 快速查找:主键索引通常是一个B+树索引,这种结构允许数据库高效地进行查找、插入和删除操作。因此,使用主键进行查询通常比使用其他索引或无索引的查询要快得多。
  5. 聚簇索引:在某些数据库系统中,主键索引默认是聚簇索引(Clustered Index)。聚簇索引决定了数据在物理存储上的顺序,这意味着按主键顺序访问数据时性能最佳。一个表只能有一个聚簇索引,因为数据只能有一种物理排序方式。
  6. 约束作用:主键不仅是一个索引,还是一种约束。它确保了表中数据的完整性和一致性,防止了重复记录的插入。
  7. 外键引用:主键通常被用作其他表的外键(Foreign Key),以建立表之间的关系。外键必须引用一个唯一的列,通常是另一个表的主键。

2.1.2 创建主键索引

  1. 创建表时在字段后面指定
    命令:

    create table t1(
    id int primary key,
    name varchar(10),
    age tinyint
    );
    
  2. 创建表时,在最后面指定某列为索引
    命令:

    create table t2(
    id int,
    name varchar(10),
    age tinyint,
    primary key(id)
    );
    
  3. 创建表结束后,用 alter命令
    命令:

    create table t3(
    id int,
    name varchar(10),
    age tinyint
    );alter table t3 add primary key(id);
    

2.2 唯一键索引

2.2.1 唯一键索引的特点

  1. 唯一性
    唯一键索引确保索引列中的每一个值都是唯一的,不允许出现重复值。这是唯一键索引的核心特性,确保了数据的唯一性和完整性。
  2. 允许多个NULL值
    与主键索引不同,唯一键索引允许列中有多个NULL值。这是因为NULL在数据库中被视为未知值,而不是具体的重复值。
  3. 提高查询性能
    唯一键索引可以显著提高查询性能,特别是在需要确保某一列或组合列的值唯一的情况下。通过创建唯一键索引,数据库可以在查询时更快地找到特定的记录。
  4. 约束作用
    唯一键索引不仅是一个索引,还是一种约束。它确保了表中数据的完整性和一致性,防止了重复记录的插入。
  5. 可以应用于多个列
    唯一键索引可以应用于单个列,也可以应用于多个列的组合。当应用于多个列的组合时,确保整个组合的值是唯一的,而不是单个列的值。

2.2.2 唯一索引的创建

  1. 创建表时在字段后面指定
    命令:

    create table t1(
    id int unique,
    name varchar(10),
    age tinyint
    );
    
  2. 创建表时,在最后面设置某列为索引
    命令:

    create table t2(
    id int,
    name varchar(10),
    age tinyint,
    unique(id)
    );
    
  3. 创建表结束后,用 alter命令
    命令:

    create table t3(
    id int,
    name varchar(10),
    age tinyint
    );alter table t3 add unique(id);
    

2.3 普通索引

2.3.1 普通索引的特点

  1. 允许重复值
    普通索引允许索引列中的值重复。这意味着同一个值可以在索引列中出现多次。
  2. 提高查询性能
    普通索引可以显著提高查询性能,特别是对于经常用于查询条件的列。通过创建索引,数据库可以更快地定位到所需的记录。
  3. 不强制非空
    普通索引不要求列中的值必须是非空的。列中的值可以是NULL,并且可以有多个NULL值。
  4. 可以应用于多个列
    普通索引可以应用于单个列,也可以应用于多个列的组合。当应用于多个列的组合时,索引会根据组合列的值进行排序和查找。
  5. 不影响数据插入和更新
    创建普通索引不会像唯一索引那样对数据插入和更新施加额外的约束。这意味着在插入或更新数据时,即使索引列中有重复值,也不会引发错误。

总的来说:普通索引的主要目的是提高查询性能,特别是在频繁用于查询条件的列上

2.3.2 普通索引的创建

  1. 创建表时,在最后面设置某列为索引
    命令:

    create table t1(
    id int,
    name varchar(10),
    age tinyint,
    index(id)
    );
    
  2. 创建表后,用alter命令
    命令:

    create table t2(
    id int,
    name varchar(10),
    age tinyint
    );alter table t2 add index(id);
    
  3. 创建表后,创建一个索引名为 idx_name 的索引,create index
    命令:

    create table t3(
    id int,
    name varchar(10),
    age tinyint
    );create index ind_t3_id on t3(id);
    

    在这里插入图片描述

2.4 全文索引

2.4.1 全文索引的作用

当对文章字段或有大量文字的字段进行检索时,会使用到全文索引。MySQL提供全文索引机制,但是有要求,要求表的存储引擎必须是MyISAM,而且默认的全文索引支持英文,不支持中文。如果对中文进行全文检索,可以使用sphinx的中文版(coreseek)。

早期版本的MySQL(5.6及之前)中,全文索引仅支持MyISAM存储引擎。
从MySQL 5.7开始,InnoDB存储引擎也支持全文索引。

全文索引的主要作用:

  1. 提高搜索性能
  • 快速查找:全文索引使用专门的算法(如倒排索引)来加速文本搜索。通过索引,数据库可以快速定位包含特定单词或短语的记录,而不需要扫描整个表。
  • 减少I/O操作:全文索引减少了磁盘I/O操作,提高了查询效率,特别是在处理大量文本数据时。
  1. 支持复杂的搜索操作
  • 模糊匹配:全文索引可以支持模糊匹配,如部分单词匹配、前缀匹配等。
  • 短语搜索:可以搜索包含特定短语的记录。
  • 近义词搜索:通过配置,可以支持近义词搜索,提高搜索的准确性和相关性。
  • 布尔搜索:支持使用布尔运算符(如AND、OR、NOT)进行复杂查询。
  1. 自然语言搜索
  • 自然语言处理:全文索引可以支持自然语言搜索,即用户可以用自然语言形式的查询语句进行搜索,系统会返回最相关的记录。
  • 权重计算:全文索引可以根据关键词在文档中的出现频率和位置等因素计算权重,返回最相关的记录。

2.4.2 全文索引的创建

创建全文索引:

CREATE TABLE articles (id INT UNSIGNED AUTO_INCREMENT NOT NULL PRIMARY KEY,title VARCHAR(200),body TEXT,FULLTEXT (title,body))engine=MyISAM;

插入数据:

 INSERT INTO articles (title,body) VALUES('MySQL Tutorial','DBMS stands for DataBase ...'),('How To Use MySQL Well','After you went through a ...'),('Optimizing MySQL','In this tutorial we will show ...'),('1001 MySQL Tricks','1. Never run mysqld as root. 2. ...'),('MySQL vs. YourSQL','In the following database comparison ...'),('MySQL Security','When configured properly, MySQL ...');

查询有没有database数据
命令:select * from articles where body like '%database%';
在这里插入图片描述
显然,可以查询出结果。

但是,有没有使用全文索引呢?
可以explain 工具查看一下。
命令:
explain select * from articles where body like '%database%';
在这里插入图片描述
下面是各项的解释

  1. id: 查询块的标识号。在这个例子中只有一个查询块,所以id为1。

  2. select_type: 表明查询的类型。这里"SIMPLE"表示这个查询只涉及一张表。

  3. table: 被访问的表名。这里是"articles"表。

  4. partitions: 如果表被分区,则此列为分区名称。在这里,由于表未分区,因此为NULL。

  5. type: 访问类型,表明MySQL如何读取数据。这里的"ALL"意味着全表扫描,即MySQL将遍历整张表的所有行。

  6. possible_keys: 可能使用的索引列表。在这个例子中,没有列出任何可能的索引,说明查询没有使用索引。

  7. key: 实际使用的索引。同样,这里也是NULL,确认了没有使用索引的事实。

  8. key_len: 索引中使用的字节数。既然没有使用索引,这一项也为NULL。

  9. ref: 显示了哪个列或常量被用来查找行。这里为NULL,再次确认没有使用索引来优化查询。

  10. rows: MySQL估计的要检查的行数。这里估计为6行。

  11. filtered: 过滤掉不符合WHERE子句的行后剩余的百分比。这里为16.67%,意味着大约只有16.67%的数据会被实际返回给查询。

  12. Extra: 包含其他额外的信息。这里的"Using where"表示MySQL正在使用WHERE子句过滤行。

总的来说,这个EXPLAIN输出表明了一个全表扫描的操作,没有使用任何索引,可能会导致较慢的查询速度,尤其是在大表上的情况。如果可能的话,考虑添加适当的索引以改善查询性能。

已经创建全文索引了,什么没有使用呢?怎么才能使用呢?
命令:

select * from articles
where match(body) against ('database');

在这里插入图片描述

为什么会报错呢?

答:当你在一个全文索引中指定了多个列时,查询时必须使用相同的列集

命令:

select * from articles 
where match(title,body) against('database');

在这里插入图片描述
explain工具查询:
在这里插入图片描述
key对应的是title,可以看出,使用的是title索引,所以,改查询使用索引了。

为什么key只是对应title而不是对应titlebody 呢?

答:创建全文索引的时候使用的是 fulltext(title,body),所以该全文索引是多列,并且该全文索引的名字是第一个列名–title 。这里的key对应的是索引的名字

3. 查询索引

  1. 方法一:
    语法:show keys from table_name;
    上述命令显示的内容看着不方便可以使用:
    show keys from table_name\G

    示例:
    命令:show keys from articles \G

    在这里插入图片描述

  2. 方法2:
    语法:show index from table_name;
    或:show index from table_name\G

    示例:
    命令:show index from t1\G
    在这里插入图片描述

  3. 方法3(查询到的信息表简略):
    语法:desc table_name;

4. 删除索引

  1. 删除主键索引
    语法:alter table table_name drop primary key;

    一个表中只用一个主键索引,像这种范式的删除方法其实还是针对它一个索引。

    示例:
    命令:alter table t1 drop primary;
    在这里插入图片描述

  2. 其他索引的删除
    语法:alter table table_name drop index column_name;
    在这里插入图片描述

  3. 使用 drop index
    语法:drop index index_name on table_name;

    该语法不能删除主键,主键是表中重要的组成部分,只能用 alter table t1 drop primary进行删除

    示例:

    删除表t1的唯一索引unique
    命令:drop index unique on t1
    在这里插入图片描述
    为什么会报错呢?
    答:unique 是索引的类型,不是索引的名字。删除索引的时候要先查询索引的类型

    查询一下索引的名字:show index from t1\G
    在这里插入图片描述

    删除number列的索引drop index number on t1;
    在这里插入图片描述

5. 索引创建原则

索引创建原则

  • 比较频繁作为查询条件的字段应该创建索引
  • 唯一性太差的字段不适合单独创建索引,即使频繁作为查询条件
  • 更新非常频繁的字段不适合作创建索引
  • 不会出现在where子句中的字段不该创建索引

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/59389.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于SpringBoot+微信小程序+协同过滤算法+二维码订单位置跟踪的农产品销售平台-新

✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取项目下载方式&#x1f345; 一、项目背景介绍&#xff1a; “农产品商城”小程序…

C++ | Leetcode C++题解之第541题反转字符串II

题目&#xff1a; 题解&#xff1a; class Solution { public:string reverseStr(string s, int k) {int n s.length();for (int i 0; i < n; i 2 * k) {reverse(s.begin() i, s.begin() min(i k, n));}return s;} };

一个由Deno和React驱动的静态网站生成器

大家好&#xff0c;今天给大家分享一个由 Deno React 驱动的静态网站生成器Pagic。 项目介绍 Pagic 是一个由 Deno React 驱动的静态网站生成器。它配置简单&#xff0c;支持将 md/tsx 文件渲染成静态页面&#xff0c;而且还有大量的官方或第三方主题和插件可供扩展。 核心…

如何才能实时监测Mac的运行状态

实时监测Mac的运行状态&#xff0c;能够让我们更好的了解Mac的情况&#xff0c;因此如何才能监测Mac的运行状态很重要 State&#xff0c;实时监测你的Mac运行状态&#xff0c;能够直观的展示当前Mac的CPU、内存、硬盘、温度、风扇、网络信息以及开机时间等重要信息 除此之外&a…

python之正则表达式总结

正则表达式 对于正则表达式的学习&#xff0c;我整理了网上的一些资料&#xff0c;希望可以帮助到各位&#xff01;&#xff01;&#xff01; 我们可以使用正则表达式来定义字符串的匹配模式&#xff0c;即如何检查一个字符串是否有跟某种模式匹配的部分或者从一个字符串中将与…

练习LabVIEW第三十八题

学习目标&#xff1a; 刚学了LabVIEW&#xff0c;在网上找了些题&#xff0c;练习一下LabVIEW&#xff0c;有不对不好不足的地方欢迎指正&#xff01; 第三十八题&#xff1a; 创建一个VI&#xff0c;实现对按钮状态的指示和按钮“按下”持续时间简单计算功能&#xff0c;按…

众测遇到的一些案列漏洞

文章中涉及的敏感信息均已做打码处理,文章仅做经验分享用途,切勿当真,未授权的攻击属于非法行为!文章中敏感信息均已做多层打码处理。传播、利用本文章所提供的信息而造成的任何直接或者间接的后果及损失,均由使用者本人负责,作者不为此承担任何责任,一旦造成后果请自行…

C#:强大而优雅的编程语言

在当今的软件开发领域&#xff0c;C#作为一种广泛应用的编程语言&#xff0c;以其强大的功能、优雅的语法和丰富的生态系统&#xff0c;受到了众多开发者的喜爱。本文将深入探讨 C#的各个方面&#xff0c;展示它的魅力和优势。 一、C#的历史与发展 C#是由微软公司开发的一种面…

【OJ题解】在字符串中查找第一个不重复字符的索引

&#x1f4b5;个人主页: 起名字真南 &#x1f4b5;个人专栏:【数据结构初阶】 【C语言】 【C】 【OJ题解】 目录 1. 引言2. 题目分析示例&#xff1a; 3. 解题思路思路一&#xff1a;双重循环思路二&#xff1a;哈希表 4. C代码实现5. 代码详解6. 时间和空间复杂度分析7. 优化方…

el-date-picker日期选择器动态设置日期

需求&#xff1a;选择开始时间&#xff0c;或者在开始时间已存在的情况下&#xff1b;结束时间下拉日期选择框展示从开始日期展示&#xff1b;而不是当前日期&#xff0c;并且结束时间下拉框日期要禁用开始时间之前的日期。 <el-form-item label"开始时间" prop&q…

「C/C++」C/C++的区别

✨博客主页何曾参静谧的博客&#x1f4cc;文章专栏「C/C」C/C程序设计&#x1f4da;全部专栏「VS」Visual Studio「C/C」C/C程序设计「UG/NX」BlockUI集合「Win」Windows程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「PK」Parasoli…

Redis-基本了解

一、Redis 初识 Redis 是⼀种基于键值对&#xff08;key-value&#xff09;的NoSQL数据库&#xff0c;与很多键值对数据库不同的是&#xff0c;Redis 中的值可以是由string&#xff08;字符串&#xff09;、hash&#xff08;哈希&#xff09;、list&#xff08;列表&#xff09…

大模型面试题持续更新_Moe(2024-10-30)

获取更多面试真题的集合&#xff0c;请移步至 https://i.afbcs.cn/naPbNYhttps://pica.zhimg.com/80/v2-7fd6e77f69aa02c34ca8c334870b3bcd_720w.webp?sourced16d100b Moe和集成学习方法有什么异同&#xff1f; MoE和集成学习的思想异曲同工&#xff0c;都是集成了多个模型的…

配置深度学习环境

先前已经配置好了 1在新建一个项目时 2.打开文件&#xff0c;找到设置 3.点开设置 如图1.2.3所示

PHP不良事件上报系统源码,医院安全不良事件管理系统,基于 vue2+element+ laravel框架开发

不良事件上报系统通过 “事前的人员知识培训管理和制度落地促进”、“事中的事件上报和跟进处理”、 以及 “事后的原因分析和工作持续优化”&#xff0c;结合预存上百套已正在使用的模板&#xff0c;帮助医院从对护理事件、药品事件、医疗器械事件、医院感染事件、输血事件、意…

Rust 力扣 - 2461. 长度为 K 子数组中的最大和

文章目录 题目描述题解思路题解代码题目链接 题目描述 题解思路 我们遍历长度为k的窗口&#xff0c;用一个哈希表记录窗口内的所有元素&#xff08;用来对窗口内元素去重&#xff09;&#xff0c;我们取哈希表中元素数量等于k的窗口总和的最大值 题解代码 use std::collecti…

yolov8涨点系列之C2f模块改进主分支

文章目录 C2F 模块介绍定义与基本原理应用场景 C2f模块修改步骤(1) C2f_up模块编辑(2)在__init_.pyblock.py中声明&#xff08;3&#xff09;在task.py中声明yolov8引入C2f_up模块yolov8.yamlyolov8.yaml引入C2f_up模块 C2f改进对YOLOv8检测具有多方面的好处 C2F 模块介绍 定义…

数字IC后端实现Innovus 时钟树综合(Clock Tree Synthesis)典型案例

对于如下所示电路&#xff0c;要求以下几路做到等长&#xff0c;clock skew控制在50ps以内&#xff0c;clock tree insertion delay做到800ps! from FF/Q to FF1_1/D through the FF1 CK from FF/Q to FF2_1/D through the FF2 CK from FF/Q to FF3_1/D through the FF3 CK fr…

STM32F030中断言的使用分享

前言 最近在写一个程序中&#xff0c;想对存到FLASH中的结构体分配的大小做控制&#xff0c;希望分配的大小偶数字节大小。&#xff08;因为读时是按16位读&#xff0c;如果奇数就可能读超了&#xff09;如果结构体大小为奇数&#xff0c;就跳到断言处。 分析 STM32F030的标…

Flutter使用share_plus是提示发现了重复的类

问题描述 我现在下载了share_plus包后发现代码编译不通过&#xff0c;并提示Duplicate class kotlin.collections.jdk8.CollectionsJDK8Kt found in modules jetified-kotlin-stdlib-1.8.22 (org.jetbrains.kotlin:kotlin-stdlib:1.8.22) and jetified-kotlin-stdlib-jdk8-1.7…