MySQL查询执行(三):显示随机消息

假设有如下表结构:

-- 创建表words
CREATE TABLE `words` (`id` int(11) NOT NULL AUTO_INCREMENT,`word` varchar(64) DEFAULT NULL,PRIMARY KEY (`id`)
) ENGINE=InnoDB;--数据生成存储过程
delimiter ;;
create procedure idata()
begindeclare i int;set i=0;while i<10000 doinsert into words(word) values(concat(char(97+(i div 1000)), char(97+(i % 1000 div 100)), char(97+(i % 100 div 10)), char(97+(i % 10))));set i=i+1;end while;
end;;
delimiter ;-- 生成数据call
call idata();

思考:假设有一个英语学习App,首页有一个随机显示单词的功能,也就是根据每个用户的级别有一个单词表,然后这个用户每次访问首页的时候,都会随机滚动显示三个单词。如果让你来设计这个SQL语句,你会怎么写呢?

内存临时表


首先, 你会想到用order by rand()来实现这个逻辑。

select word from words order by rand() limit 3;

这个语句的意思很直白, 随机排序取前3个。 虽然这个SQL语句写法很简单, 但执行流程却有点复杂的。

先用explain命令来看看这个语句的执行情况。

Extra字段显示Using temporary, 表示的是需要使用临时表; Using filesort, 表示的是需要执行排序操作。

因此这个Extra的意思就是, 需要临时表, 并且需要在临时表上排序。

问:对于临时内存表的排序来说, 它会选择哪一种算法呢?

对于InnoDB表来说, 执行全字段排序会减少磁盘访问, 因此会被优先选择。

对于内存表(Memory表), 回表过程只是简单地根据数据行的位置, 直接访问内存得到数据, 根本不会导致多访问磁盘。

优化器没有了这一层顾虑, 那么它会优先考虑的, 就是用于排序的行越少越好了, 所以, MySQL这时就会选择rowid排序。

上述SQL语句执行流程如下:

1)创建一个临时表。 这个临时表使用的是memory引擎, 表里有两个字段, 第一个字段是double类型, 为了后面描述方便, 记为字段R, 第二个字段是varchar(64)类型, 记为字段W。 并且, 这个表没有建索引。

2)从words表中, 按主键顺序取出所有的word值。 对于每一个word值, 调用rand()函数生成一个大于0小于1的随机小数, 并把这个随机小数和word分别存入临时表的R和W字段中, 到此, 扫描行数是10000。

3)现在临时表有10000行数据了, 接下来你要在这个没有索引的内存临时表上, 按照字段R排序。

4)初始化 sort_buffer。 sort_buffer中有两个字段, 一个是double类型, 另一个是整型。

5)从内存临时表中一行一行地取出R值和位置信息(我后面会和你解释这里为什么是“位置信息”) , 分别存入sort_buffer中的两个字段里。 这个过程要对内存临时表做全表扫描, 此时扫描行数增加10000, 变成了20000。

6)在sort_buffer中根据R的值进行排序。 注意, 这个过程没有涉及到表操作, 所以不会增加扫描行数。

7)排序完成后, 取出前三个结果的位置信息, 依次到内存临时表中取出word值, 返回给客户端。 这个过程中, 访问了表的三行数据, 总扫描行数变成了20003。

接下来, 我们通过慢查询日志(slow log) 来验证一下我们分析得到的扫描行数是否正确。

# Query_time: 0.900376 Lock_time: 0.000347 Rows_sent: 3 Rows_examined: 20003 
SET timestamp=1541402277;
select word from words order by rand() limit 3;

其中, Rows_examined: 20003就表示这个语句执行过程中扫描了20003行, 也就验证了我们分析得出的结论。

上述SQL语句执行示意图如下:

问1:SQL执行流程中,从内存临时表中一行一行地取出R值和位置信息,此处位置信息是指什么?

每个引擎都有用来唯一标识数据行的信息,InnoDB使用rowid来唯一标识行数据,MEMORY引擎使用位置信息来唯一标识行数据(MEMORY引擎不是索引组织表)。可以把MEMORY表理解为一个数组,而位置信息即为数组的下标。

问2:如果把一个InnoDB表的主键删掉, 是不是就没有主键, 就没办法回表了?

如果你创建的表没有主键, 或者把一个表的主键删掉了, 那么InnoDB会自己生成一个长度为6字节的rowid来作为主键。

1)对于有主键的InnoDB表来说, 这个rowid就是主键ID。

2)对于没有主键的InnoDB表来说, 这个rowid就是由系统生成的。

3)MEMORY引擎不是索引组织表。 在这个例子里面, 你可以认为它就是一个数组。 因此, 这个rowid其实就是数组的下标。

磁盘临时表


问1:是不是所有的临时表都是内存表?

其实不是的。 tmp_table_size这个配置限制了内存临时表的大小, 默认值是16M。 如果临时表大小超过了tmp_table_size, 那么内存临时表就会转成磁盘临时表。

注:内存临时表使用的引擎默认是MEMORY;磁盘临时表使用的引擎默认是InnoDB, 是由参数internal_tmp_disk_storage_engine控制的。

现在有如下场景:

--设置语句执行环境;
set tmp_table_size=1024;
set sort_buffer_size=32768;
set max_length_for_sort_data=16;/* 打开 optimizer_trace,只对本线程有效*/
SET optimizer_trace='enabled=on'; /* 执行语句 */
select word from words order by rand() limit 3;/* 查看 OPTIMIZER_TRACE输出*/
SELECT * FROM `information_schema`.`OPTIMIZER_TRACE`\G

执行结果:

因为将max_length_for_sort_data设置成16, 小于word字段的长度定义, 所以我们看到sort_mode里面显示的是rowid排序, 这个是符合预期的, 参与排序的是随机值R字段和rowid字段组成的行。

问2:R字段存放的随机值就8个字节, rowid是6个字节(至于为什么是6字节, 就留给你课后思考吧) , 数据总行数是10000, 这样算出来就有140000字节,超过了sort_buffer_size 定义的 32768字节了。 但是, number_of_tmp_files的值居然是0, 难道不需要用临时文件吗?

答:采用的是MySQL 5.6版本引入的一个新的排序算法,即: 优先队列排序算法(即堆排序)。 而非归并排序。

因为现在的SQL语句,只需要取R值最小的3个rowid。 如果使用归并排序算法的话, 虽然最终也能得到前3个值, 但是这个算法结束后, 已经将10000行数据都排好序了,显然浪费了非常多的计算量。

而优先队列算法, 就可以精确地只得到三个最小值, 执行流程如下:

1)对于这10000个准备排序的(R,rowid), 先取前三行, 构造成一个堆;

2)取下一个行(R’,rowid’), 跟当前堆里面最大的R比较, 如果R’小于R, 把这个(R,rowid)从堆中去掉, 换成(R’,rowid’);

3)重复第2步, 直到第10000个(R’,rowid’)完成比较。

优先队列排序过程示意图:

上图模拟6个(R,rowid)行, 通过优先队列排序找到最小的三个R值的行的过程。 整个排序过程中, 为了最快地拿到当前堆的最大值, 总是保持最大值在堆顶, 因此这是一个最大堆。

上图OPTIMIZER_TRACE结果中, filesort_priority_queue_optimization这个部分的chosen=true, 就表示使用了优先队列排序算法, 这个过程不需要临时文件, 因此对应的number_of_tmp_files是0。

这个流程结束后, 我们构造的堆里面, 就是这个10000行里面R值最小的三行。 然后, 依次把它们的rowid取出来, 去临时表里面拿到word字段, 这个过程就跟上一篇文章的rowid排序的过程一样了。

问3:针对“order by是怎么工作的”小节中的SQL查询语句:

select city,name,age from t where city='杭州' order by name limit 1000 ;

这里也用到了limit, 为什么没用优先队列排序算法呢?

答:这条SQL语句是limit 1000, 如果使用优先队列算法的话, 需要维护的堆的大小就是1000行的(name,rowid), 超过了我设置的sort_buffer_size大小, 所以只能使用归并排序算法。

结论:不论是使用哪种类型的临时表, order by rand()这种写法都会让计算过程非常复杂, 需要大量的扫描行数, 因此排序过程的资源消耗也会很大。

随机排序方法


随机算法1

思考:如果只随机选择1个word值, 可以怎么做呢?

答:思路上是这样的:

1)取得这个表的主键id的最大值M和最小值N。

2)用随机函数生成一个最大值到最小值之间的数 X = (M-N)*rand() + N。

3)取不小于X的第一个ID的行。

我们把这个算法暂时称作随机算法1。上述思路语句执行过程:

--1)取得这个表的主键id的最大值M和最小值N;
select max(id), min(id) into @M, @N from t;--2)用随机函数生成一个最大值到最小值之间的随机数;
set @X= floor((@M - @N + 1)*rand() + @N);--3)取不小于X的第一个ID的行;
select * from t where id >= @X limit 1;

这个方法效率很高, 因为取max(id)和min(id)都是不需要扫描索引的, 而第三步的select也可以用索引快速定位, 可以认为就只扫描了3行。

但实际上, 这个算法本身并不严格满足题目的随机要求, 因为ID中间可能有空洞, 因此选择不同行的概率不一样, 不是真正的随机。

比如你有4个id, 分别是1、 2、 4、 5, 如果按照上面的方法, 那么取到 id=4的这一行的概率是取得其他行概率的两倍。

如果这四行的id分别是1、 2、 40000、 40001呢? 这个算法基本就能当bug来看待了。

随机算法2

为了得到严格随机的结果,对上述流程优化后如下:

1)取得整个表的行数, 并记为C。

2)取得 Y = floor(C * rand())。 floor函数在这里的作用, 就是取整数部分。

3)再用limit Y,1取得一行。

我们把这个算法, 称为随机算法2。 下面这段代码, 就是上面流程的执行语句的序列。

select count(*) into @C from t;-- rand()返回0到1之间的一个随机数
-- rand(5)返回0-5之间的一个随机数
set @Y = floor(@C * rand());-- concat()函数用于拼接字符串
set @sql = concat("select * from t limit ", @Y, ", 1");
prepare stmt from @sql;
execute stmt;
DEALLOCATE prepare stmt;

由于limit 后面的参数不能直接跟变量, 所以我在上面的代码中使用了prepare+execute的方法。

这个随机算法2, 解决了算法1里面明显的概率不均匀问题。(使用limit替换>=)

MySQL处理limit Y,1 的做法就是按顺序一个一个地读出来, 丢掉前Y个, 然后把下一个记录作为返回结果, 因此这一步需要扫描Y+1行。 再加上, 第一步扫描的C行, 总共需要扫描C+Y+1行,执行代价比随机算法1的代价要高。

当然, 随机算法2跟直接order by rand()比起来, 执行代价还是小很多的。

思考:如果按照这个表有10000行来计算的话, C=10000, 要是随机到比较大的Y值, 那扫描行数也跟20000差不多了, 接近order by rand()的扫描行数, 为什么说随机算法2的代价要小很多呢?

因为随机算法2进行limit获取数据的时候是根据主键排序获取的,主键天然索引排序。获取到第9999条的数据也远比order by rand()方法的组成临时表,对R字段排序再获取rowid代价小的多。(order by rand()方法有临时表、R字段排序等消耗)

随机算法3

思考:如果我们按照随机算法2的思路, 要随机取3个word值呢? 应该怎么做?

答:

1)取得整个表的行数, 记为C。

2)根据相同的随机方法得到Y1、 Y2、 Y3。

3)再执行三个limit Y, 1语句得到三行数据。

我们把这个算法, 称作随机算法3。下面这段代码, 就是上面流程的执行语句的序列。

select count(*) into @C from t;set @Y1 = floor(@C * rand());
set @Y2 = floor(@C * rand());
set @Y3 = floor(@C * rand());select * from t limit @Y1, 1;    //在应用代码里面取Y1、Y2、Y3值,拼出SQL后执行
select * from t limit @Y2, 1;
select * from t limit @Y3, 1;

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

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

相关文章

[工具]GitHub + PicGo 搭建免费博客图床

文章目录 起因GitHub新建GitHub仓库新建token授予picgo权限 PicGOPicGO上传失败原因 起因 还是觉得个人博客记录最好还是不要money&#x1f625;&#xff0c;所以还是想白嫖&#xff0c;找到了GitHub PicGO的方式&#xff0c;记录一下。 GitHub 过程和搭建博客链接类似&…

DDOS攻击学习 - kali初学

文章目录 本地ssh配置nmap(网络连接的工具)nmap -sP IP地址nmap -p 1-65535 -A IP地址主机发现Ping扫描端口扫描时序扫描常用扫描方式指纹识别与探测全端口版本探测防火墙/IDS逃逸报文分段信息收集IP信息收集WHOIS查询数据库渗透测试MySQL列举数据库列举MySQL变量发起请求目录扫…

PostgreSQL的pg-collector工具

PostgreSQL的pg-collector工具 pg-collector 是一个用于 PostgreSQL 数据库的监控和数据收集工具。它主要用于收集 PostgreSQL 实例的性能指标、查询统计和日志信息&#xff0c;以便进行数据库性能分析和故障排查。通过收集这些数据&#xff0c;管理员可以更好地了解数据库的运…

day3 测试基础知识

1. 你认为性能测试的目的是什么&#xff1f;做好性能测试的工作的关键是什么&#xff1f; 性能测试工作的目的是检查系统是否满足在需求说明书中规定的性能&#xff0c;性能测试常常需要和强度测试结合起来&#xff0c;并常常要求同时进行软件和硬件的检测。 性能测试主要的关…

关于SpringBoot项目利用阿里EasyExcel快捷导入Excel文件入库初始化数据的简单实现

一、问题描述 无论新项目还是旧项目&#xff0c;都会出现数据维护、数据初始化等操作&#xff0c;手动录显然很low(领导会骂你)&#xff0c;所以一般采用批量导入导出。这里你还在用原始读取excel逐行逐列去读取吗&#xff1f;2024了ok&#xff1f;利用工具是我们cv大师的一贯…

在 OpenEuler24.03 源码安装 PG16.3

在ANOLIS 23上源码安装了16.1&#xff0c;在OpenEuler24.03上PG16.3&#xff0c;安装也是一样的吗&#xff1f; 抱着这样的态度&#xff0c;我试​着去安装&#xff0c;如果不关闭SELINUX&#xff0c;还是有一个差异的&#xff0c;同时&#xff0c;发现即使是最小安装&#xf…

ElasticSearch学习篇15_《检索技术核心20讲》进阶篇之TopK检索

背景 学习极客实践课程《检索技术核心20讲》https://time.geekbang.org/column/article/215243&#xff0c;文档形式记录笔记。 相关问题&#xff1a; ES全文检索是如何进行相关性打分的&#xff1f;ES中计算相关性得分的时机?如何加速TopK检索&#xff1f;三种思路 精准To…

GEE:设置ui.Map.Layer上交互矢量边界填充颜色为空,只显示边界

一、目标 最近在GEE的交互功能鼓捣一些事情&#xff0c;在利用buffer功能实现了通过选点建立一个矩形后&#xff0c;需要将该矩形填充颜色设为空&#xff0c;只留边界。 然而通过正常设置layer的可视化参数并不能实现这一目的。因此只能另辟蹊径&#xff0c;改为定义矢量边界…

【JavaEE精炼宝库】 网络编程套接字——初识网络编程 | UDP数据报套接字编程

文章目录 一、网络编程基础1.1 网络编程的意义&#xff1a;1.2 网络编程的概念&#xff1a;1.3 网络编程的术语解释&#xff1a;1.4 常见的客户端服务端模型&#xff1a; 二、Socket 套接字2.1 Socket 套接字的概念&#xff1a;2.2 Socket 套接字的分类&#xff1a; 三、UDP数据…

fetchApi === 入门篇

目录 fetch 基本认知 fetch 如何使用 Response对象&#xff08;了解&#xff09; 常见属性 常见方法 fetch 配置参数 fetch发送post请求 fetch 函数封装 fetch 实战 - 图书管理案例 渲染功能 添加功能 删除数据 完整代码 fetch 基本认知 思考&#xff1a; 以前开发…

NumpyPandas:Pandas库(25%-50%)

目录 前言 一、列操作 1.修改变量列 2.筛选变量列 3.删除变量列 4.添加变量列 二、数据类型的转换 1.查看数据类型 2.将 ok的int类型转换成float类型 3.将ar的float类型转换成int类型 三、建立索引 1.建立DataFrame时建立索引 2.在读入数据时建立索引 3.指定某列或…

virtualbox ubuntu扩充磁盘大小

首先在虚拟存储管理里面修改磁盘大小 然后安装gparted sudo gparted 打开管理工具 选中要调整的区域右键选择调整区域大小 拖动上述位置就可以实现扩容。完成后点击应用 然后重启虚拟机即可。

基于深度学习技术及强大的专家团队,针对多个工业垂类场景进行算法优化的智慧城管开源了。

智慧城管视觉监控平台是一款功能强大且简单易用的实时算法视频监控系统。它的愿景是最底层打通各大芯片厂商相互间的壁垒&#xff0c;省去繁琐重复的适配流程&#xff0c;实现芯片、算法、应用的全流程组合&#xff0c;从而大大减少企业级应用约95%的开发成本。 基于深度学习技…

微服务的入门

带着问题进行学习&#xff1a; 1. 对服务进行拆分后&#xff0c;物理上是隔离的&#xff0c;数据上也是隔离的&#xff0c;如何进行不同服务之间进行访问呢&#xff1f; 2.前端是怎么样向后端发送请求的&#xff1f; 通过http请求&#xff0c;通过url&#xff0c;请求的…

wireshark--流量分析利器

&#x1f3bc;个人主页&#xff1a;金灰 &#x1f60e;作者简介:一名简单的大一学生;易编橙终身成长社群的嘉宾.✨ 专注网络空间安全服务,期待与您的交流分享~ 感谢您的点赞、关注、评论、收藏、是对我最大的认可和支持&#xff01;❤️ &#x1f34a;易编橙终身成长社群&#…

传输层(port)UDP/TCP——解决怎么发,发多少,出错了怎么办

**传输层&#xff1a;**负责数据能够从发送端传输接收端. 传输层所封装的报头里一定有&#xff1a;源端口号和目的端口号的。 **端口号&#xff1a;**可以标识一台主机中的唯一一个进程&#xff08;运用程序&#xff09;&#xff0c;这样当数据传输到传输层的时候就可以通过端…

LangChain开发框架并学会对大型预训练模型进行微调(fine-tuning)

要掌握LangChain开发框架并学会对大型预训练模型进行微调&#xff08;fine-tuning&#xff09;&#xff0c;你需要理解整个过程从数据准备到最终部署的各个环节。下面是这一流程的一个概览&#xff0c;并提供了一些关键步骤和技术点&#xff1a; 1. LangChain开发框架简介 La…

Maven实战(一)- Maven安装与配置

Maven实战&#xff08;一&#xff09;- Maven安装与配置 文章目录 Maven实战&#xff08;一&#xff09;- Maven安装与配置1.下载安装包2.配置环境变量。3.安装目录分析4.设置HTTP代理5.镜像 前言&#xff1a; ​ 最近博主看完了《Maven实战》&#xff08;许晓斌著&#xff09;…

回调函数地狱及其解决方法——Promise链式调用以及async函数和await

一、什么是回调函数地狱&#xff1f; > 在回调函数一直向下嵌套回调函数&#xff0c;形成回调函数地狱 二、回调函数地狱问题&#xff1f; > 可读性差 > 异常捕获困难 > 耦合性严重 <!DOCTYPE html> <html lang"en"><head>&…

大数据时代,区块链是如何助力数据开放共享的?

在大数据时代&#xff0c;区块链技术以其独特的优势&#xff0c;为数据开放共享提供了强有力的支持。以下是区块链助力数据开放共享的几个主要方面&#xff1a; 1. 增强数据安全性与隐私保护 加密安全&#xff1a;区块链技术采用先进的加密算法&#xff0c;如国密非对称加密技…