mysql学习之select查询语句到底是怎么执行的?

本篇文章给大家带来了关于mysql中select语句执行的相关知识,其中包括了连接器、分析器、优化器和执行器,希望对大家有帮助。

mysql学习之select查询语句到底是怎么执行的?

mysql 作为一个关系型数据库,在国内使用应该是最广泛的。也许你司使用 Oracle、Pg 等等,但是大多数互联网公司,比如我司使用得最多的还是 Mysql,重要性不言而喻。

执行select * from table,数据库底层到底发生了啥?从而我们得到数据呢?

假设现在我有张 user 表,只有两列,一列 id 自增的,一列 name 是  varchar 类型。建表语句是这样的:

CREATE TABLE IF NOT EXISTS `user`(   `id` INT UNSIGNED AUTO_INCREMENT,   `name` VARCHAR(100) NOT NULL,   PRIMARY KEY ( `id` ))ENGINE=InnoDB DEFAULT CHARSET=utf8;

问题就是下面这个语句的执行过程。

select * from user where id = 1;

01 mysql 架构概览

要想理解这个问题就必须要知道 mysql 的内部架构。为此,我画了张 mysql 的架构图(你也可以理解为 sql 查询语句的执行过程),如下所示:

36.png

首先 msql 分为 server 层和存储引擎层两个部分。server 层包括四个功能模块,分别是:连接器、查询缓存、优化器、执行器。这一层负责了 mysql 的所有核心工作,比如:内置函数、存储过程、触发器以及视图等。

而存储引擎层则是负责数据的存取。注意,存储引擎在 mysql 是可选的,常见的还有:InnoDB、MyISAM 以及 Memory等,最常用的就是 InnoDB。现在默认的存储引擎也是它(从 mysql 5.5.5 版本开始),大家可以看到我上面的建表语句就是指定了 InnoDB 引擎。当然,你不指定的话默认也是它。

由于存储引擎是可选的,所以 mysql 中,所有的存储引擎其实是共用一个 server层的。回到正题,我们就以这张图的流程来解决一下小胖的问题。

1.1 连接器

首先,数据库要执行 sql,肯定要先连接数据库吧。这部分工作就是由连接器完成。它负责校验账户密码、获取权限、管理连接数,最终与客户端建立连接等工作。mysql 链接数据库是这样写的:

mysql -h 127.0.0.1 -P 3306 -u root -p# 127.0.0.1 : ip 3306 : 端口 root : 用户名

运行命令之后需要输入密码,当然也可以跟在 -p 后面。不过不建议这么做,会有密码泄露的风险。

输入命令后,连接器根据你的账户名密码验证身份。这是会出现两种情况:

账号或密码不对,服务端会返回一个 “ERROR 1045 (28000): Access denied for user ‘root’@’127.0.0.1’ (using password: YES)” 的错误,退出连接。

验证通过,连接器就会到权限表查出你的权限。之后你有啥权限都要通过这时读到的权限进行判断。

注意,我说的是此时查到的权限。就算你用管理员账号修改了当前用户的权限,此时已连接上的当前用户不受影响,必须要重启 mysql 新的权限才会生效。

1.1.1 查看连接状态

连接完成,如果后续没有做任何事情,这个连接就处于空闲状态。你可以用 show processlist; 命令查看 mysql 的连接信息,如下图,我的数据库连接都是 Sleep 状态的,除了执行 show processlist 操作的连接。

01.png

1.1.2 控制连接

如果客户端太长时间没有操作,此连接将会自动断开。这个时间默认是 8 小时,由参数 wait_timeout 控制。如果断开以后继续操作就会收到 “Lost connection to MySQL server during query”的错误。这时就必须重连才能执行请求。

数据库里面有长短连接之分,长连接:连接成功后不断有请求,就会一直使用同一连接。短连接:每次执行完几次请求就断开连接,下次需要再建立。

由于建立连接是比较耗时的操作,所以建议使用长连接。但这会有个问题长连接一直连着就会导致内存占用过大,被系统强行沙雕。从而导致 MySQL 异常重启。如何解决呢?两个方法:

定期断开长连接。使用特定时间,或者程序判断执行一个占用内存大的操作后,断开连接。之后需要操作就重连。

mySQL 5.7 或以上版本,可以在每次执行一个占用内存大的操作后,执行mysql_reset_connection来重新连接资源,此时不需重连或重新做权限认证,但会把连接状态恢复到刚创建完时。

1.2 查询缓存

连接建立以后可以执行 select 语句了。这就会来到第二步:查询缓存。

查询缓存中存储的数据是 key-value 的形式,key 是查询语句,value 是查询的结果。逻辑是这样的:先看看查询缓存有没该语句对应的 value?有则直接取出返回客户端,无则继续到数据库执行语句。查出结果后会放一份到缓存中,再返回客户端。

你可能发现缓存真的香,但是并不建议使用查询缓存,因为有弊端。查询缓存的失效非常频繁,只有某个表有更新。它马上失效了,对于经常更新的表来说,命中缓存的概率极低。它仅仅适用于那些不经常更新的表。

而 MySQL 似乎也考虑到这点了。提供了 query_cache_type 参数,把它设置为 DEMAND 就不再适用韩村。而对于要使用缓存的语句则可用 SQL_CACHE 显示指定,像这样:

select SQL_CACHE * from user where id = 1;

PS:MySQL 8.0 及以上版本把查询缓存删掉了,之后再也没有这块功能了。

1.3 分析器

如果没有命中缓存就进入分析器,这里就是对 sql 进行分析。分析器会做词法分析。你输入的 sql 是啥,由啥组成,MySQL 都需要知道它们代表什么。

首先根据 “select” 识别出这是查询语句。字符串”user”识别成”表名 user”、字符串”id”识别成”列名id”。

之后进行语法分析,它会根据输入的语句分析是不是符合 MySQL 的语法。具体表现就是 select、where、from 等关键字少了个字母,明显不符合 MySQL 语法,这次就会报个语法错误的异常:它一般会提示错误行数,关注”use near”后面即可。

02.png

1.4 优化器

过了分析器,就来到了优化器。MySQL 是个聪明的仔,再执行之前会自己优化下客户端传过来的语句,看看那种执行起来不那么占内存、快一点。比如下面的 sql 语句:

select * from user u inner join role r on u.id = r.user_id where u.name = "狗哥" and r.id = 666

它可以先从 user 表拿出 name = “狗哥” 记录的 ID 值再跟 role 表内连接查询,再判断 role 表里面 id 的值是否 = 666

也可以反过来:先从 role 表拿出 id = 666 记录的 ID 值再跟 user 表内连接查询,在判断 user 表里面的 name 值是否 = “狗哥”。

两种方案的执行结果是一样的,但是效率不一样、占用的资源也就不一样。优化器就是在选择执行的方案。它优化的是索引应该用哪个?多表联查应该先查哪个表?怎么连接等等。

1.5 执行器

分析器知道了做啥、优化器知道了应该怎么做。接下来就交给执行器去执行了。

开始执行,判断是否有相应的权限。比如该账户对 user 表没权限就返回无权限的错误,如下所示:

select * from user where id = 1;ERROR 1142 (42000): SELECT command denied to user 'nasus'@'localhost' for table 'user'

PS:如果命中缓存没走到执行器这里,那么在返回查询结果时做权限验证。

回到正题,如果有权限,继续打开表执行。执行器会根据表定义的引擎去使用对应接口。比如我们上面的 sql 语句执行流程是这样的:

走 id 索引、调用 InnoDB 引擎取”满足条件的第一行”接口,再循环调用”满足条件的下一行”接口(这些接口都是存储引擎定义好的),直到表中不再有满足条件的行。执行器就将上述遍历得到的行组成结果集返回给客户端。

对于 id 不是索引的表,执行器只能调用”取表记录的第一行”接口,再判断 id 是否 = 1。如果不是则跳过,是则存在结果集中;再调存储引擎接口取”下一行”,重复判断逻辑,直到表的最后一行。

至此,整个 SQL 的执行流程完毕,

推荐学习:mysql视频教程

以上就是mysql学习之select查询语句到底是怎么执行的?的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/180597.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月1日 07:12:33
下一篇 2025年11月1日 07:13:19

相关推荐

  • c++如何连接MySQL数据库_c++ MySQL数据库连接方法

    使用MySQL Connector/C++连接数据库需先安装开发库,配置编译环境并链接相应库文件,然后通过X DevAPI或C API建立连接。推荐使用X DevAPI进行现代C++开发,示例代码展示了如何连接、执行查询及处理异常,也可选用经典C API实现更底层控制。 要在C++中连接MySQL数…

    2025年12月19日
    000
  • C++如何实现简易登录注册系统

    答案是文件存储因无需额外配置、使用标准库即可操作且便于理解,成为C++简易登录注册系统的首选方式。其核心在于通过fstream读写文本文件,用简单结构体存储用户信息,注册时检查用户名唯一性并追加数据,登录时逐行比对凭据,适合初学者掌握基本I/O与逻辑控制。 C++实现简易登录注册系统,通常我们会采用…

    2025年12月18日
    000
  • C++如何实现简易问卷调查程序

    答案是C++简易问卷程序通过定义问题结构、用户交互和文件存储实现,支持文本与单选题,利用枚举区分类型,结构体存储数据,fstream保存结果,可扩展为多态设计以增强灵活性和可维护性。 C++实现一个简易的%ignore_a_1%程序,核心思路其实不复杂:你需要定义好问卷的结构,比如每个问题长什么样,…

    2025年12月18日
    000
  • 工厂模式在C++中怎样应用 简单工厂与抽象工厂对比

    简单工厂通过参数决定创建何种产品,适用于产品少且变化少的场景;抽象工厂则通过接口创建相关产品族,支持扩展而不修改代码,适合复杂系统。 工厂模式在C++中主要用于解耦对象的创建与使用,提升代码的可维护性和扩展性。根据复杂度和应用场景的不同,常见的有简单工厂和抽象工厂两种形式。它们都能实现对象的动态创建…

    2025年12月18日
    000
  • C++如何实现文件操作的回滚机制 事务性文件处理设计

    c++++中实现文件操作的回滚机制,其核心在于手动构建“事务性”保障,以确保数据的一致性和完整性。1. 回滚机制的本质是通过预留恢复路径(如临时文件、日志记录等),在操作失败时将文件状态还原至修改前;2. 与数据库事务的区别在于,数据库内置acid特性支持原子性、一致性、隔离性和持久性,而文件系统无…

    2025年12月18日 好文分享
    000
  • 如何用C++编写快递管理系统 物流状态追踪和数据库基础

    高效的物流状态更新机制设计可通过消息队列实现异步处理。首先,使用消息队列(如rabbitmq或kafka)解耦状态更新服务与核心业务逻辑,在状态变化时发送消息至队列;其次,由消费者服务异步处理并批量更新数据库,以降低频繁更新对数据库的压力。 快递管理系统的核心在于追踪物流状态和高效管理数据。C++虽…

    2025年12月18日 好文分享
    000
  • C++工业自动化测试环境怎么配置 LabVIEW与C++混合编程

    c++++与labview混合编程的核心在于分工明确、高效协作,c++负责高性能计算和底层硬件控制,labview用于界面设计与系统集成;具体步骤包括搭建c++开发环境并生成dll、配置labview开发环境、设计清晰的接口、使用“调用库函数节点”调用c++ dll,并注意数据类型映射、内存管理、调…

    2025年12月18日 好文分享
    000
  • C++智慧农业物联网环境怎么搭建 LoRaWAN网关开发配置

    搭建基于c++++的智慧农业物联网环境并配置lorawan网关的核心步骤如下:1. 准备硬件,选择兼容的lorawan模块和网关,并确保网络连接;2. 配置网关软件,设置频率、服务器地址等参数;3. 使用c++开发后端服务,接收并解析数据,存入数据库并提供api;4. 可选chirpstack搭建本…

    2025年12月18日 好文分享
    000
  • 如何用C++实现桥接模式 抽象与实现分离设计方案

    c++++中桥接模式的核心优势在于解耦抽象与实现,使其能独立变化。1. 它通过将一个类中可能变动的具体操作抽离为独立的实现体系,降低类组合数量,避免“m x n”组合爆炸;2. 抽象类(如shape)包含指向实现接口的指针或引用,调用具体实现(如drawingapi),使两者互不影响;3. 适用于多…

    2025年12月18日 好文分享
    000
  • C++中如何使用类型擦除_运行时多态实现

    c++++中类型擦除是一种在运行时统一处理不同类型的技术,通过隐藏具体类型信息实现手动多态。1. 定义抽象基类作为通用接口;2. 创建模板类实现该接口并转发操作;3. 使用包装类包含模板类实例指针,提供相同方法并转发调用。示例中drawable为抽象基类,circle和square为具体类型,dra…

    2025年12月18日 好文分享
    000
  • 静态检查融合:SonarQube + Clang实现C++自动审计

    静态检查融合是通过结合sonarqube规则引擎与c++lang分析能力提升c++代码审计效果的方法。1. 安装sonarqube服务器并配置数据库;2. 安装sonarscanner并配置环境变量;3. 安装clang及相关开发工具;4. 安装并配置sonarqube cfamily插件;5. 创…

    2025年12月18日 好文分享
    000
  • 什么是C++中的数据库索引优化?

    c++++中的数据库索引优化可以通过以下步骤实现:1)选择正确的索引类型,如b-tree适合范围查询;2)维护索引,平衡读写性能;3)使用复合索引提升多列查询性能;4)通过查询优化和性能监控调整索引策略。 C++中的数据库索引优化?这是一个有趣且复杂的主题,让我们深入探讨一下。 C++虽然主要用于系…

    2025年12月18日
    000
  • C++中的ORM框架是什么?

    c++++中的orm框架是一种工具,通过对象映射到数据库表,简化数据库操作。1)orm框架通过对象操作数据库,提高开发效率和代码可维护性。2)常见框架如cppdb、odb和soci,各有特点。3)使用时需注意映射关系、查询优化和性能考虑。 C++中的ORM(对象关系映射)框架是什么?简单来说,ORM…

    2025年12月18日
    000
  • C语言网络编程中数据库连接的优化措施问答

    优化措施:使用连接池预创建数据库连接,避免连接/断开开销。优化查询语句,只查询必要字段,避免通配符或子查询。使用批处理将多个操作组合成一个请求,减少网络开销。尽量使用乐观的锁定机制,减少锁定粒度。考虑使用无服务器数据库,由供应商管理数据库,减少开销。 C语言网络编程中数据库连接的优化措施 在C语言网…

    2025年12月18日
    000
  • C++ 函数库在哪些场景下使用?

    c++++ 函数库预先定义了代码模块,可用于处理常见任务,广泛应用于:输入/输出操作字符串操作数学运算此外,还有特定领域的场景,如:图形处理网络编程数据库连接示例:使用 matplotlibcpp 函数库绘制折线图,实现了便捷的图形化展示。 C++ 函数库的应用场景 简介C++ 函数库提供了预定义的…

    2025年12月18日
    000
  • C++框架内置功能在跨平台开发中的作用

    在跨平台开发中,c++++ 框架的内置功能发挥着至关重要的作用,包括:跨平台兼容性:在多平台上稳定运行,简化移植。图形用户界面 (gui) 支持:提供跨平台的 gui 库,无需编写平台特定代码。数据库集成:支持多种数据库系统,实现数据访问和操作的跨平台性。网络通信:提供网络通信机制,用于分布式应用程…

    2025年12月18日
    000
  • C++框架中的数据处理和存储机制

    本文介绍了 c++++ 框架中处理和存储数据的关键技术:数据处理:stl 容器(vector、map、set)boost 库(额外容器和算法)eigen 库(优化矩阵和向量)数据存储:关系数据库管理系统(rdbms)(mysql、postgresql、sqlite)文档存储(mongodb、redi…

    2025年12月18日
    000
  • C++框架在行业中的实际案例

    c++++ 框架广泛应用于行业中,提供高效且可维护的代码解决方案。这些框架用于开发跨平台 gui(qt)、科学计算(armadillo、eigen)、线性代数运算(eigen)、计算机视觉(opencv)和高性能网络应用程序(ace)。例如,spotify 使用 boost(线程管理和网络操作)、a…

    2025年12月18日
    000
  • C++框架在后端开发方面的优缺点?

    c++++框架在后端开发中的优缺点:优点:高性能低内存使用跨平台原生数据库支持丰富的库生态缺点:学习曲线陡峭内存管理复杂缺乏动态关注编译时间长缺乏垃圾回收 C++ 框架在后端开发中的优缺点 优点: 高性能: C++ 是一种编译语言,可生成高效机器码,从而实现优异的性能。低内存使用: C++ 提供直接…

    2025年12月18日
    000
  • 在线资源和教程如何帮助降低 C++ 框架的学习难度?

    通过在线资源和教程,你可以降低 c++++ 框架的学习难度:利用官方文档和外部教程学习框架的基本概念和最佳实践。加入 stack overflow 和讨论论坛等社区,寻求具体问题的帮助和支持。通过构建 restful api、数据库应用程序和网络服务器等实战案例,获得实践经验。 通过在线资源和教程降…

    2025年12月18日
    000

发表回复

登录后才能评论
关注微信