当前位置: 首页 > news >正文

MySQL Hints:控制查询优化器的选择

码到三十五 : 个人主页

MySQL Hints是优化数据库查询性能的一种强大工具。它们允许开发者在SQL查询中嵌入指令,以影响MySQL优化器的决策过程。在某些情况下,优化器可能无法选择最佳的查询执行计划,这时我们可以使用Hints来引导优化器做出更好的选择。

目录

    • 一、什么是MySQL Hints
    • 二、为什么需要使用Hints
    • 三、如何使用Hints
      • 1. 确定需要使用的Hint
      • 2. 编写Hint注释
      • 3. 将Hint注释与SQL语句结合
      • 4. 测试和验证
      • 语法说明
    • 四、常用的MySQL Hints
      • 1. `USE INDEX` 和 `FORCE INDEX`
      • 2. `IGNORE INDEX`
      • 3. `STRAIGHT_JOIN`
      • 4. `SQL_NO_CACHE`
      • 5. `INDEX_MERGE` 和 `NO_INDEX_MERGE`
      • 6. **`JOIN_FIXED_ORDER`**
      • 7. **`BLOCK_NESTED_LOOP`**, **`BATCHED_KEY_ACCESS`**, **`NO_BNL`**, 和 **`NO_BKA`**
      • 8. **`MRR`** 和 **`NO_MRR`**
      • 9. **`FILESORT`** 和 **`NO_FILESORT`**
      • 10. **`SUBQUERY`** 和 **`NO_SUBQUERY`**
      • 11. **`DERIVED_MERGE`** 和 **`NO_DERIVED_MERGE`**
    • 五、优化器Hints与`optimizer_switch`的区别
    • 六、使用Hints的注意事项
    • 七、结语

一、什么是MySQL Hints

MySQL Hints是一组特殊的注释或指令,可以直接嵌入到SQL查询中,以改变MySQL优化器的默认行为。这些Hints通常被用于解决性能问题,或者当开发者比优化器更了解数据分布和查询特性时,来指导优化器选择更好的查询计划。

在这里插入图片描述

二、为什么需要使用Hints

  1. 性能调优:在某些复杂的查询场景下,优化器可能无法自动选择最优的执行计划。通过Hints,我们可以手动指定一些执行策略,从而提升查询性能。

  2. 控制执行计划:当数据库中的数据分布或表结构发生变化时,优化器可能会选择不同的执行计划。使用Hints可以确保查询的稳定性,即使在数据或表结构发生变化时,也能保持相同的执行计划。

  3. 解决特定问题:有时,我们可能会遇到一些特定的问题,如索引选择不当、连接顺序不佳等。Hints提供了一种快速解决问题的方法,而无需更改表结构或重写查询。

三、如何使用Hints

Hints是通过在SQL语句前添加特殊格式的注释来使用的。通常的格式是/*+ HintName(parameters) */。这些Hints只对紧跟其后的SQL语句有效,并且不会影响其他查询。以下是如何在SQL语句中使用Hints的详细步骤:

1. 确定需要使用的Hint

首先,你需要确定你想要使用的Hint。这通常基于你对查询性能的分析和对MySQL优化器行为的理解。例如,如果你发现优化器没有选择你认为最优的索引,你可能会想要使用FORCE INDEXIGNORE INDEX等Hints。

2. 编写Hint注释

在SQL语句之前,你需要添加一个特殊格式的注释来包含你的Hint。这个注释的格式是/*+ HintName(parameters) */,其中HintName是你想要使用的Hint的名称,parameters是该Hint所需的任何参数。

例如,如果你想要强制优化器使用特定的索引,可以这样写:

/*+ FORCE INDEX(table_name idx_name) */

在这里,table_name是你想要应用Hint的表的名称,而idx_name是你想要强制优化器使用的索引的名称。

3. 将Hint注释与SQL语句结合

一旦你编写了Hint注释,你需要将它放在SQL语句之前,并确保它们之间没有换行或其他字符。这样,优化器就能识别并应用你的Hint。

一个完整的带有Hint的SQL查询像这样:

/*+ FORCE INDEX(my_table my_index) */ SELECT * FROM my_table WHERE my_column = 'value';

在这个例子中,FORCE INDEX Hint告诉优化器在执行查询时强制使用my_table上的my_index索引。

4. 测试和验证

在应用了Hint之后,你应该测试查询以确保Hint产生了预期的效果。你可以使用EXPLAIN语句来查看查询的执行计划,并确认优化器是否按照你的Hint来执行查询。

EXPLAIN /*+ FORCE INDEX(my_table my_index) */ SELECT * FROM my_table WHERE my_column = 'value';

这将显示查询的执行计划,并允许你验证FORCE INDEX Hint是否已被正确应用。

语法说明

值得注意的是,/*+ … */ 这种注释语法是Oracle数据库中的一种标准方式来提供优化器hints,但在MySQL中,这种语法并不是官方的。在MySQL中,你通常不需要使用特殊的注释语法来提供FORCE INDEX hint。相反,你可以直接在查询中使用它,如下所示:

SELECT * FROM my_table FORCE INDEX (my_index) WHERE my_column = 'value';

FORCE INDEX (my_index) 直接与SELECT语句结合,告诉MySQL优化器在执行查询时强制使用my_index索引。这是MySQL支持的标准语法,而不需要使用特殊的注释格式。

总结来说,FORCE INDEX 必须与查询语句一起使用,而不是作为一个独立的语句执行。在MySQL中,你不需要使用/*+ … */注释语法来提供这个hint,而是可以直接在查询中指定。如果你在使用其他数据库系统(如Oracle),那么可能需要使用该系统的特定注释语法来提供优化器hints。

四、常用的MySQL Hints

以下是对一些常用的MySQL Hints的详细介绍以及相应的代码:

1. USE INDEXFORCE INDEX

这两个Hints用于指定查询时要使用的索引。USE INDEX是建议性的,而FORCE INDEX更为强制。

-- USE INDEX 示例
SELECT * FROM users USE INDEX (idx_age) WHERE age > 30;-- FORCE INDEX 示例
SELECT * FROM users FORCE INDEX (idx_age) WHERE age > 30;

在上述示例中,我们指示MySQL在查询users表时优先使用idx_age索引。

2. IGNORE INDEX

这个Hint用于指示MySQL在查询时忽略指定的索引。

SELECT * FROM users IGNORE INDEX (idx_age) WHERE name = 'John Doe';

在这个示例中,我们告诉MySQL在执行查询时忽略idx_age索引。

3. STRAIGHT_JOIN

STRAIGHT_JOIN用于强制MySQL按照指定的表顺序进行JOIN操作,而不是由优化器自动选择。

SELECT * FROM users STRAIGHT_JOIN orders ON users.id = orders.user_id;

在这个示例中,我们强制MySQL先扫描users表,然后再与orders表进行JOIN。

4. SQL_NO_CACHE

这个Hint用于指示MySQL不使用查询缓存,确保每次查询都直接访问数据库。

SELECT SQL_NO_CACHE * FROM users WHERE age > 30;

在这个示例中,我们确保查询结果不是从缓存中获取的,而是直接查询数据库。

5. INDEX_MERGENO_INDEX_MERGE

这两个Hints影响优化器是否使用索引合并策略。

-- INDEX_MERGE 示例(鼓励使用索引合并)
SELECT * FROM users INDEX_MERGE (idx_age, idx_name) WHERE age = 30 OR name = 'John Doe';-- NO_INDEX_MERGE 示例(阻止使用索引合并)
SELECT * FROM users NO_INDEX_MERGE WHERE age = 30 OR name = 'John Doe';

INDEX_MERGE示例中,我们鼓励优化器考虑合并idx_ageidx_name索引来加速查询。在NO_INDEX_MERGE示例中,我们阻止优化器使用索引合并。

6. JOIN_FIXED_ORDER

  • 作用:强制MySQL按照查询中指定的表顺序进行JOIN操作,不进行顺序的优化调整。
SELECT * FROM table1 JOIN_FIXED_ORDER JOIN table2 ON table1.id = table2.table1_id;

7. BLOCK_NESTED_LOOP, BATCHED_KEY_ACCESS, NO_BNL, 和 NO_BKA

  • 这些Hints影响JOIN操作的执行策略。
-- BLOCK_NESTED_LOOP 示例
SELECT * FROM users a BLOCK_NESTED_LOOP JOIN orders b ON a.id = b.user_id;-- BATCHED_KEY_ACCESS 示例
SELECT * FROM users a BATCHED_KEY_ACCESS JOIN orders b ON a.id = b.user_id;-- NO_BNL 示例
SELECT * FROM users a NO_BNL JOIN orders b ON a.id = b.user_id;-- NO_BKA 示例
SELECT * FROM users a NO_BKA JOIN orders b ON a.id = b.user_id;

8. MRRNO_MRR

  • MRR 作用:鼓励优化器使用多范围读取优化。
  • NO_MRR 作用:阻止优化器使用多范围读取优化。
-- MRR 示例
SELECT * FROM users WHERE id IN (1, 3, 5) PROCEDURE ANALYSE() MRR;-- NO_MRR 示例
SELECT * FROM users WHERE id IN (1, 3, 5) PROCEDURE ANALYSE() NO_MRR;

注意PROCEDURE ANALYSE() 是一个诊断过程,通常与 MRRNO_MRR 一起使用来分析和优化查询,但它在实际应用中并不常见。

9. FILESORTNO_FILESORT

-- 强制使用文件排序
SELECT * FROM users ORDER BY age FILESORT;-- 阻止使用文件排序(尽管这通常不是推荐的,因为优化器通常会选择最佳方法)
SELECT * FROM users ORDER BY age NO_FILESORT;

10. SUBQUERYNO_SUBQUERY

-- 鼓励优化器保留子查询
SELECT * FROM users WHERE id IN (SELECT user_id FROM orders WHERE amount > 100) SUBQUERY;-- 鼓励优化器不使用子查询,可能转换为JOIN操作
SELECT * FROM users WHERE id IN (SELECT user_id FROM orders WHERE amount > 100) NO_SUBQUERY;

11. DERIVED_MERGENO_DERIVED_MERGE

-- 鼓励优化器合并派生表
SELECT * FROM (SELECT * FROM users WHERE age > 25) AS derived1 DERIVED_MERGE JOIN orders ON derived1.id = orders.user_id;-- 阻止优化器合并派生表
SELECT * FROM (SELECT * FROM users WHERE age > 25) AS derived1 NO_DERIVED_MERGE JOIN orders ON derived1.id = orders.user_id;

优化器的Hints是MySQL中一种特殊的注释语法,用于向查询优化器提供关于如何执行SQL查询的建议或指令。这些Hints为开发者提供了一种机制,以便在必要时能够更精细地控制查询的执行计划,尤其是在优化器自动选择的计划不是最优的情况下。

五、优化器Hints与optimizer_switch的区别

  • optimizer_switch:这是一个系统变量,通过它可以开启或关闭某些优化器的特性或策略。改变这个变量会影响所有后续的查询执行。因此,如果你需要对不同的查询应用不同的优化策略,你需要在每个查询之前更改optimizer_switch,这在实际操作中可能会很不方便。

  • 优化器Hints:与optimizer_switch不同,优化器Hints允许你在单个SQL语句中指定优化策略。这种方法提供了更精细的控制,因为你可以针对每个查询或查询中的特定表指定不同的优化策略。此外,语句中的Hints会覆盖optimizer_switch的设置。

六、使用Hints的注意事项

  1. 谨慎使用:过度或不当地使用Hints可能会导致性能下降,因为它们可能会覆盖优化器的智能决策。

  2. 测试和验证:在应用Hints之前和之后,都要对查询性能进行彻底的测试,以确保它们确实带来了预期的提升。

  3. 版本兼容性:不是所有的MySQL版本都支持所有的Hints,因此在使用前要检查你的MySQL版本是否支持所需的Hints。

  4. 可维护性:在SQL查询中嵌入Hints可能会降低代码的可读性和可维护性。确保团队成员都了解并同意使用这些Hints。

  5. 监控和调优:即使使用了Hints,也应该定期监控查询性能,并根据需要进行调整。

七、结语

MySQL Hints是一种强大的工具,可以帮助我们解决复杂的查询性能问题。然而,它们应该谨慎使用,并且总是与彻底的测试和验证相结合。通过正确使用Hints,我们可以引导MySQL优化器做出更明智的决策,从而提高数据库查询的性能和稳定性。

参考: https://dev.mysql.com/doc/refman/8.0/en/controlling-optimizer.html


听说...关注下面公众号的人都变牛了,纯技术,纯干货 !

相关文章:

MySQL Hints:控制查询优化器的选择

码到三十五 : 个人主页 MySQL Hints是优化数据库查询性能的一种强大工具。它们允许开发者在SQL查询中嵌入指令,以影响MySQL优化器的决策过程。在某些情况下,优化器可能无法选择最佳的查询执行计划,这时我们可以使用Hints来引导优化…...

【TB作品】msp430g2553单片机,OLED,PCF8591,ADC,DAC

硬件 OLED PCF8591 /** OLED* VCC GND* SCL接P2^0* SDA接P2^1*//** PCF8591* VCC GND* SCL接P1^4* SDA接P1^5*//* 板子上按键 P1.3 *//* 单片机ADC输入引脚 P1.1 *//* 说明:将PCF8591的DAC输出接到单片机ADC输入引脚 P1.1,单片机采集电压并显示 */功能…...

C#WPF数字大屏项目实战10--不良指标分页

1、区域划分 2、区域布局 3、视图模型 4、控件绑定 5、运行效果 走过路过,不要错过,欢迎点赞,收藏,转载,复制,抄袭,留言,动动你的金手指,财务自由...

数字塔问题

#include<iostream> using namespace std; //从下向上得到最优值 void dtower(int a[][100],int s[][100],int n) {for(int in; i>1; i--){for(int j1; j<i; j){if(in)s[i][j]a[i][j];else{int ts[i1][j];if(t<s[i1][j1])ts[i1][j1];s[i][j]a[i][j]t;}}} } void…...

【介绍下Pwn,什么是Pwn?】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…...

Python:b站多个视频爬取下载

&#x1f4da;博客主页&#xff1a;knighthood2001 ✨公众号&#xff1a;认知up吧 &#xff08;目前正在带领大家一起提升认知&#xff0c;感兴趣可以来围观一下&#xff09; &#x1f383;知识星球&#xff1a;【认知up吧|成长|副业】介绍 ❤️如遇文章付费&#xff0c;可先看…...

Java常规题技术分享

一、数组排序和添加成员 设计类Student和类StudentClass。 (1) 类Student有字符串属性name、double属性grade和int属性age 有带参数的构造方法&#xff0c;可设置三个属性的值 有各个属性的置取方法 (2)类StudentClass有Student数组属性stus存放班级成员&#xff0c;有int…...

Pytorch语义分割(1)-----加载数据

在语义分割中用到的数据无非就是原始图片&#xff08;image&#xff09;和标注后得到的mask图片&#xff0c;所以在读取数据的时候只要返回图片和标签信息就OK 了。 import torch import os import numpy as np from torch.utils.data import Dataset from utils_func import …...

Java中加号的多种用途

在Java中&#xff0c; 符号有多种用途&#xff0c;主要根据上下文而定。以下是在Java中的一些主要用途&#xff1a; 加法运算符&#xff1a; 这是最常见的用途&#xff0c;用于数字相加。 int a 5;int b 3;int sum a b; // sum is 8 字符串连接符&#xff1a; 当用…...

React useCallback用法

useCallback 是 React 中的一个 Hook&#xff0c;它用于优化性能&#xff0c;通过缓存函数的引用来避免在组件的每次渲染时都创建新的函数实例。这对于避免不必要的子组件重新渲染特别有用&#xff0c;因为如果传递给子组件的回调函数在每次渲染时都不同&#xff0c;即使子组件…...

Flutter 中的 ErrorWidget 小部件:全面指南

Flutter 中的 ErrorWidget 小部件&#xff1a;全面指南 Flutter 是一个由 Google 开发的跨平台 UI 框架&#xff0c;它允许开发者使用 Dart 语言构建高性能、美观的应用。在 Flutter 的丰富组件库中&#xff0c;ErrorWidget 是一个特殊的组件&#xff0c;用于在渲染过程中捕获…...

【数据结构】穿梭在二叉树的时间隧道:顺序存储的实现

专栏引入 哈喽大家好&#xff0c;我是野生的编程萌新&#xff0c;首先感谢大家的观看。数据结构的学习者大多有这样的想法&#xff1a;数据结构很重要&#xff0c;一定要学好&#xff0c;但数据结构比较抽象&#xff0c;有些算法理解起来很困难&#xff0c;学的很累。我想让大家…...

【数据结构与算法 经典例题】链表的回文结构(图文详解)

&#x1f493; 博客主页&#xff1a;倔强的石头的CSDN主页 &#x1f4dd;Gitee主页&#xff1a;倔强的石头的gitee主页 ⏩ 文章专栏&#xff1a;《数据结构与算法 经典例题》C语言 期待您的关注 ​ 目录 一、问题描述 二、解题思路 三、C语言代码实现 一、问题描述 二、解…...

通过DirectML和ONNXRuntime运行Phi-3模型

更多精彩内容&#xff0c;欢迎关注我的公众号“ONE生产力”&#xff01; 上篇我们讲到通过Intel Core Ultra系列处理器内置的NPU加速运行Phi-3模型&#xff0c;有朋友评论说他没有Intel处理器是否有什么办法加速Phi-3模型。通常&#xff0c;使用GPU特别是NVIDA的GPU加速AI模型…...

C语言经典例题-18

1.判断是不是字母 题目描述: KK想判断输入的字符是不是字母&#xff0c;请帮他编程实现。 输入描述: 多组输入&#xff0c;每一行输入一个字符。 输出描述: 针对每组输入&#xff0c;输出单独占一行&#xff0c;判断输入字符是否为字母&#xff0c;输出内容详见输出样例。 输…...

计算机网络之crc循环冗余校验、子网划分、rip协议路由转发表、时延计算、香浓定理 奈氏准则、TCP超时重传 RTO

crc循环冗余校验 异或运算 : 相同得0,相异得1 从多项式获取除数 在原数据的末端补0 , 0的个数等于最高次项的阶数 如果最后结果的有效位数较少时&#xff0c;前面应该补0&#xff0c;补到个数与阶位相同 子网划分 子网掩码&#xff1a;用于识别IP地址中的网络号和主机号的…...

揭秘高效人事财务对接新方案!

一、客户介绍 某生物医药科技有限公司是一家专注于生物创新药物研发与生产的科技型企业。公司的主要业务范围包括技术开发、技术服务、医学研究与试验发展、经济信息咨询、企业管理等。公司凭借其强大的技术实力、丰富的研发经验和优秀的团队阵容&#xff0c;在生物创新药领域…...

Unity中的MVC框架

基本概念 MVC全名是Model View Controller 是模型(model)-视图(view)-控制器(controller)的缩写 是一种软件设计规范&#xff0c;用一种业务逻辑、数据、界面显示 分离的方法组织代码 将业务逻辑聚集到一个部件里面&#xff0c;在改进和个性化定制界面及用户交互的同时&#x…...

网工内推 | 上市公司网工,Base广东,思科DE/IE认证优先

01 广州赛意信息科技股份有限公司 &#x1f537;招聘岗位&#xff1a;技术架构师 &#x1f537;职责描述&#xff1a; 1、设计、开发和维护工业数据库及其架构&#xff0c;包括数据采集、存储、处理和分析的工具和系统。 2、开发和维护数据管道和工作流程&#xff0c;确保数据…...

ZYNQ AXI4 FDMA内存读写

1 概述 如果用过ZYNQ的都知道,要直接操作PS的DDR 通常是DMA 或者VDMA,然而用过XILINX 的DMA IP 和 VDMA IP,总有一种遗憾,那就是不够灵活,还需要对寄存器配置,真是麻烦。对于我们搞 FPGA 的人来说,最喜欢直接了当,直接用FPGA代码搞定。现在XILINX 的总线接口是AXI4总线…...

R语言AI模型部署方案:精准离线运行详解

R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...

DAY 47

三、通道注意力 3.1 通道注意力的定义 # 新增&#xff1a;通道注意力模块&#xff08;SE模块&#xff09; class ChannelAttention(nn.Module):"""通道注意力模块(Squeeze-and-Excitation)"""def __init__(self, in_channels, reduction_rat…...

vue3+vite项目中使用.env文件环境变量方法

vue3vite项目中使用.env文件环境变量方法 .env文件作用命名规则常用的配置项示例使用方法注意事项在vite.config.js文件中读取环境变量方法 .env文件作用 .env 文件用于定义环境变量&#xff0c;这些变量可以在项目中通过 import.meta.env 进行访问。Vite 会自动加载这些环境变…...

GC1808高性能24位立体声音频ADC芯片解析

1. 芯片概述 GC1808是一款24位立体声音频模数转换器&#xff08;ADC&#xff09;&#xff0c;支持8kHz~96kHz采样率&#xff0c;集成Δ-Σ调制器、数字抗混叠滤波器和高通滤波器&#xff0c;适用于高保真音频采集场景。 2. 核心特性 高精度&#xff1a;24位分辨率&#xff0c…...

稳定币的深度剖析与展望

一、引言 在当今数字化浪潮席卷全球的时代&#xff0c;加密货币作为一种新兴的金融现象&#xff0c;正以前所未有的速度改变着我们对传统货币和金融体系的认知。然而&#xff0c;加密货币市场的高度波动性却成为了其广泛应用和普及的一大障碍。在这样的背景下&#xff0c;稳定…...

React---day11

14.4 react-redux第三方库 提供connect、thunk之类的函数 以获取一个banner数据为例子 store&#xff1a; 我们在使用异步的时候理应是要使用中间件的&#xff0c;但是configureStore 已经自动集成了 redux-thunk&#xff0c;注意action里面要返回函数 import { configureS…...

并发编程 - go版

1.并发编程基础概念 进程和线程 A. 进程是程序在操作系统中的一次执行过程&#xff0c;系统进行资源分配和调度的一个独立单位。B. 线程是进程的一个执行实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位。C.一个进程可以创建和撤销多个线程;同一个进程中…...

python爬虫——气象数据爬取

一、导入库与全局配置 python 运行 import json import datetime import time import requests from sqlalchemy import create_engine import csv import pandas as pd作用&#xff1a; 引入数据解析、网络请求、时间处理、数据库操作等所需库。requests&#xff1a;发送 …...

Ubuntu系统多网卡多相机IP设置方法

目录 1、硬件情况 2、如何设置网卡和相机IP 2.1 万兆网卡连接交换机&#xff0c;交换机再连相机 2.1.1 网卡设置 2.1.2 相机设置 2.3 万兆网卡直连相机 1、硬件情况 2个网卡n个相机 电脑系统信息&#xff0c;系统版本&#xff1a;Ubuntu22.04.5 LTS&#xff1b;内核版本…...

Tauri2学习笔记

教程地址&#xff1a;https://www.bilibili.com/video/BV1Ca411N7mF?spm_id_from333.788.player.switch&vd_source707ec8983cc32e6e065d5496a7f79ee6 官方指引&#xff1a;https://tauri.app/zh-cn/start/ 目前Tauri2的教程视频不多&#xff0c;我按照Tauri1的教程来学习&…...