MySQL 8.0 SQL优化黑科技,面试官都不一定知道!
liuian 2025-07-06 14:06 58 浏览
前言
提到SQL优化,大多数人想到的还是那些经典套路:建索引、避免全表扫描、优化JOIN顺序…这些确实是基础,但如果你还停留在MySQL 5.7时代的优化思维,那就out了。
MySQL 8.0已经发布好几年了,带来了大量革命性的新特性。然而令人意外的是,很多开发者(甚至包括一些面试官)对这些新功能还不够了解,依然在用老思路解决新问题。
今天我们就来聊聊MySQL 8.0那些鲜为人知的优化黑科技。这些特性不仅能让你的SQL性能飞起,更重要的是,当你在面试中展示这些技巧时,很可能连面试官都会被惊艳到。毕竟,谁不喜欢学到点新东西呢?
如果你的项目还在用MySQL 5.7或更早版本,这篇文章也值得收藏。
因为升级到MySQL 8.0后,你就能立刻用上这些强大的功能了。
耐心看完,你一定有所收获。
正文
老套路快速回顾
在展示黑科技之前,我们先快速过一遍那些“传统艺能”:
- 索引优化:B+树索引、复合索引、覆盖索引
- 查询重写:避免SELECT *、合理使用LIMIT
- 连接优化:INNER JOIN vs LEFT JOIN,驱动表选择
- 子查询优化:EXISTS vs IN,避免相关子查询
- 表设计优化:字段类型选择、范式vs反范式
这些基础知识相信大家都太熟悉了。
现在让我们来看看MySQL 8.0带来的那些降维打击式的新玩法。
骚操作一:利用MySQL 8.0的窗口函数进行复杂查询优化
传统的分组查询往往需要多次子查询,性能堪忧。MySQL 8.0引入的窗口函数可以优雅地解决这个问题。
传统写法(需要多次扫描表):
-- 查询每个部门薪资最高的前3名员工
SELECT * FROM employees e1
WHERE (
SELECT COUNT(DISTINCT e2.salary)
FROM employees e2
WHERE e2.department = e1.department
AND e2.salary >= e1.salary
) <= 3;窗口函数优化(只需一次扫描):
WITH ranked_employees AS (
SELECT *,
ROW_NUMBER() OVER (PARTITION BY department ORDER BY salary DESC) as rn
FROM employees
)
SELECT * FROM ranked_employees WHERE rn <= 3;这种写法不仅代码更清晰,执行效率也大幅提升。
骚操作二:反直觉的索引设计 - 降序索引的妙用
MySQL 8.0开始支持真正的降序索引。很多人以为ORDER BY col DESC用普通索引就够了,但在某些场景下,降序索引能带来意想不到的效果。
-- 创建混合排序的索引
CREATE INDEX idx_mixed ON orders (status ASC, created_time DESC);
-- 这个查询现在可以完全利用索引顺序,避免filesort
SELECT * FROM orders
WHERE status = 'pending'
ORDER BY created_time DESC
LIMIT 10;特别是在分页查询中,这种索引设计能显著减少排序开销。
骚操作三:Generated Column的隐藏威力
Generated Column(生成列,又叫虚拟列)不仅仅是用来存储计算结果,还能用来进行一些曲线救国的优化。
关于虚拟列的具体使用可以看这篇文章:MySQL虚拟列:一个被低估的MySQL特性
-- 为JSON字段的某个属性创建生成列和索引
ALTER TABLE user_profiles
ADD COLUMN age_generated INT GENERATED ALWAYS AS (JSON_EXTRACT(profile_data, '$.age')) STORED,
ADD INDEX idx_age (age_generated);
-- 现在可以高效查询JSON内的数据
SELECT * FROM user_profiles WHERE age_generated BETWEEN 25 AND 35;这个方式其实特别适合处理JSON字段查询慢的问题。
骚操作四:Invisible Index - 安全的索引测试
在生产环境中,如果想给某个表加索引,又会担心新索引会影响其他SQL的执行计划,或者不确定这个索引是否真的有效果。
MySQL 8.0 的隐形索引(Invisible Index)就是为了解决这个问题而生:
-- 1. 先创建一个"隐形"的索引,这时索引会被创建,但优化器看不到它
CREATE INDEX idx_user_age ON users (age) INVISIBLE;
-- 2. 在测试会话中"激活"这个隐形索引
SET SESSION optimizer_switch = 'use_invisible_indexes=on';
-- 3. 测试查询效果
EXPLAIN SELECT * FROM users WHERE age BETWEEN 25 AND 35;
-- 这时能看到是否使用了新索引
-- 4. 如果效果好,就让所有人都能看到这个索引
ALTER INDEX idx_user_age ON users VISIBLE;
-- 5. 如果效果不好,直接删除,对线上业务无影响
-- DROP INDEX idx_user_age ON users;这样可以避免在生产环境中直接创建索引可能带来的风险。
骚操作五:巧用Hint强制执行计划
有时候优化器的选择并不是最优的,搞不好就会自作聪明,选择了一个看似合理但实际很慢的执行计划,这时候可以用Hint来调教一下。
看这个例子:
-- 假设你有这样一个查询
SELECT * FROM orders o
JOIN customers c ON o.customer_id = c.id
WHERE o.status = 'pending' AND c.city = 'Shanghai';
-- MySQL可能选择:
-- 1. 先扫描orders表找status='pending'的记录(假设有10万条)
-- 2. 再去JOIN customers表
-- 但你通过分析发现,Shanghai的客户只有1000个,应该:
-- 1. 先扫描customers表找city='Shanghai'的记录(1000条)
-- 2. 再去JOIN orders表那这时候就可以用 Hint 来指定索引:
-- 强制让customers表作为驱动表(小表驱动大表)
SELECT /*+ STRAIGHT_JOIN */ *
FROM customers c
JOIN orders o ON o.customer_id = c.id
WHERE c.city = 'Shanghai' AND o.status = 'pending';
-- 或者强制使用特定索引
SELECT /*+ INDEX(o idx_status) */ *
FROM orders o
WHERE o.status = 'pending';但是这种方式对于SQL的认知要求比较高,谨慎使用。
骚操作六:Resource Group - 查询级别的资源控制
如果你的数据库上既有在线业务查询(要求快速响应),也有数据分析查询(可以慢一点但很耗资源)时,传统做法是分开部署,但成本比较高。
-- 创建一个专门给批处理任务用的资源组
-- VCPU = 0-1 表示只能使用CPU的0号和1号核心
-- THREAD_PRIORITY = -10 表示线程优先级较低
CREATE RESOURCE GROUP batch_group
TYPE = USER
VCPU = 0-1
THREAD_PRIORITY = -10;
-- 创建一个给在线业务用的资源组(高优先级)
CREATE RESOURCE GROUP online_group
TYPE = USER
VCPU = 2-7
THREAD_PRIORITY = 0;
-- 当你要跑一个大的统计查询时
SET RESOURCE GROUP batch_group;
SELECT COUNT(*), AVG(amount)
FROM orders
WHERE created_time >= '2023-01-01'; -- 这个查询可能要跑很久
-- 在线业务查询自动使用默认或online_group
SELECT * FROM orders WHERE id = 12345; -- 这个查询不受影响这样可以避免大查询影响在线业务,大查询再也不会抢占在线业务的CPU资源,一定程度上能保证系统的响应速度。
骚操作七:巧用LATERAL JOIN解决复杂关联
有些查询用传统JOIN很难实现,或者需要写复杂的子查询。
假设你要查询每个用户最近购买的3件商品信息。
传统写法(复杂且性能差) :
SELECT u.username, u.email,
(SELECT product_name FROM orders o1 JOIN products p1 ON o1.product_id = p1.id
WHERE o1.user_id = u.id ORDER BY o1.created_time DESC LIMIT 1) as latest_product,
(SELECT product_name FROM orders o2 JOIN products p2 ON o2.product_id = p2.id
WHERE o2.user_id = u.id ORDER BY o2.created_time DESC LIMIT 1 OFFSET 1) as second_latest,
(SELECT product_name FROM orders o3 JOIN products p3 ON o3.product_id = p3.id
WHERE o3.user_id = u.id ORDER BY o3.created_time DESC LIMIT 1 OFFSET 2) as third_latest
FROM users u;LATERAL JOIN写法(简洁且高效) :
SELECT u.username, u.email, recent_orders.*
FROM users u
JOIN LATERAL (
SELECT p.product_name, o.created_time, o.amount
FROM orders o
JOIN products p ON o.product_id = p.id
WHERE o.user_id = u.id -- 关键:这里可以引用左表u的字段
ORDER BY o.created_time DESC
LIMIT 3
) recent_orders ON TRUE;LATERAL JOIN可以让右表引用左表的列,解决一些传统JOIN难以处理的场景。
骚操作八:Multi-Valued Index - 为JSON数组优化
MySQL 8.0.17引入的多值索引,专门为JSON数组查询设计:
-- 为JSON数组创建多值索引
CREATE INDEX idx_tags ON articles ((CAST(tags->'$[*]' AS CHAR(50) ARRAY)));
-- 高效查询包含特定标签的文章
SELECT * FROM articles WHERE JSON_OVERLAPS(tags, '["技术", "MySQL"]');结尾
MySQL 8.0的这些新特性,代表着SQL优化正在进入一个新时代。
这些功能不仅让我们的代码更优雅,性能也更强劲。
遗憾的是,很多公司和开发者还没有充分利用这些新特性。
如果你能在项目中合理运用这些技巧,不仅能解决传统方案难以处理的问题,还能在技术分享和或者面试过程中展现出你对新技术的敏锐度。
当然,新特性虽好,也要结合实际场景使用,并不是所有的优化都需要用到这些高级功能。
有时候一个简单的索引也能很好地解决问题。
技术的魅力在于选择合适的工具来解决合适的问题。
如果你的项目还在用老版本MySQL,不妨考虑升级到8.0。这些新特性也许能带来性能、开发效率和代码质量的整体提升。
最后想说,技术永远在进步,保持学习的心态才能在这个行业中走得更远。
今天的黑科技,可能就是明天的常规操作。
相关推荐
-
- 电脑无法从u盘启动怎么办(电脑无法从u盘启动解决方法)
-
电脑的进入不了u盘启动的解决方法:一、我们第一步需要确定的是你的u盘在别的电脑上检查一下U盘是否可读,如果可读的话是否成功制作了u盘启动盘了,因为想要启动进入pe的话需要u盘具备启动的功能。 二、如果你检查好自己的u盘已经成功制作了启动盘...
-
2026-01-13 10:05 liuian
- cpu频率越高越好吗(cpu频率越高速度越快吗)
-
高好。CPU的频率是影响CPU的一个重要因素,直观上来说,频率的高低影响了CPU的性能。频率越高,CPU性能越好;不过需要注意的是,CPU的主频表示在CPU内数字脉冲信号震荡的速度,与CPU实际的运算...
- 注册表清理软件(注册表清理软件残留软件)
-
你好!关于注册表清理工具的推荐,以下是几个值得推荐的工具:1.CCleaner:这是一款功能强大的免费清理工具,可以有效地清理注册表、垃圾文件等,使用简单方便。2.WiseRegistryCl...
- 显卡驱动升级有好处吗(显卡驱动升级有什么坏处)
-
显卡的新版本驱动能修改一些游戏,图形显示的BUG,所以新版本的显卡驱动能有效的利用显卡的资源,提高游戏性能。不仅可以修正旧版本中的BUG,而且可以进一步挖掘显卡硬件的功能,使得部分硬件功能得以充分发挥...
- w7旗舰版系统安装无线网卡(win7系统安装无线网卡)
-
要在Windows7中安装无线网卡,请按照以下步骤进行操作:1.检查您的计算机是否已安装无线网卡。您可以通过右键单击“我的电脑”并选择“属性”来查看计算机的硬件设置。如果计算机没有内置无线网卡,则...
- 腾达路由器管理员密码是什么
-
1、旧版本的腾达路由器,默认的用户名和密码都是:admin。?旧版腾达路由器的初始密码是:admin2、目前腾达新推出的无线路由器,在出厂状态下,是没有初始管理员密码的。?新版腾达路由器没有初始密码新...
- 电脑开机只有一个鼠标箭头黑屏
-
解决方法如下:1、同时按“ctrl+shlft+exc”键,调出任务管理器。2、点击任务管理器左下角的“详细信息”。3、然后点击左上角“文件”里的“运行新任务”。4、弹出新窗口,输入“explorer...
- 把vx好友删了想找回聊天记录
-
没有啦,联系人列表里没有了,聊天记录就没有了,无法进行恢复,收不到好友消息微信删除好友时会同时删除与该联系人的聊天记录,不过对方还是有双方的微信聊天记录的,删除好友后将无法发送消息给对方,所以伙伴们在...
- 163邮箱密码正确就是登不上(163邮箱密码一直错误)
-
邮箱不能登录或登录异常的原因有很多种哦,如您浏览器“隐私”或“安全”级别设置过高,或用户名、密码输入不正确、较长时间未登录被冻结等都会导致不能登录或登录异常。请您先检查一下哦。解决无法登录的方法有:...
- 移动硬盘维修费用大概是多少钱
-
芯片不需要多少钱,但数据恢复就另当别论了。。。如果认识人就帮你换个芯片板,要不了多少钱,如果是硬盘盒的芯片板坏了你就乾脆换个盒子,80左右。如果是硬盘芯片坏了,那就不好办了,没人愿意给你换阿。。。但如...
- windows资源管理器停止工作是什么原因
-
1.在进行重装系统之前,可以先检测一下windows资源管理器停止工作的原因是什么。如果是因为电脑的文件太多了,垃圾堆积导致的停止工作,我们就不需要进行重装系统。我们只需要下载一个360卫士或者其他可...
- 联想电脑24小时维修热线电话
-
1.打开Think.lenovo.com.cn网页,点击登陆。 2.输入用户名密码,点击登陆。 3.点击右上角的:返回个性化首页。 4.点击“咨询与报修”中的“网上报修”。 ...
- u盘上的系统怎么安装到电脑上
-
如果这个u盘是已经制作成为启动盘,可以进入pe系统的话就可以从u盘启动进入到pe系统中进行系统安装!如果你的意思是u盘里直接是操作系统的话,那就在bios设置里直接设定为u盘启动就好了!也可以在pe中...
- 20年前老笔记本改造升级(比较老的笔记本电脑改装)
-
答:10年前的笔记本电脑升级改造的方法。1.减少电脑后台程序。电脑和手机也是差不多的,有些软件在关闭之后并没有真正的退出,而是在后台偷偷的运行,这样也是占电脑内存,这样会导致电脑变得越来有。2....
- 住房公积金贷款计算器(住房公积金贷款计算器在线)
-
房贷、公积金贷款计算器基本养老保险金计算器基本医疗保险金计算器工伤保险计算器住房公积金缴存计算器养老保险退休金计算器五险一金及税后工资计算器失业保险计算器住房公积金贷款利息怎么计算,具体如下:公积金贷...
- 一周热门
-
-
飞牛OS入门安装遇到问题,如何解决?
-
如何在 iPhone 和 Android 上恢复已删除的抖音消息
-
Boost高性能并发无锁队列指南:boost::lockfree::queue
-
大模型手册: 保姆级用CherryStudio知识库
-
用什么工具在Win中查看8G大的log文件?
-
如何在 Windows 10 或 11 上通过命令行安装 Node.js 和 NPM
-
威联通NAS安装阿里云盘WebDAV服务并添加到Infuse
-
Trae IDE 如何与 GitHub 无缝对接?
-
idea插件之maven search(工欲善其事,必先利其器)
-
如何修改图片拍摄日期?快速修改图片拍摄日期的6种方法
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
