我的编程空间,编程开发者的网络收藏夹
学习永远不晚

MySQL怎样选择合适的索引

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

MySQL怎样选择合适的索引

这篇文章主要介绍了MySQL怎样选择合适的索引,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

先来看一个栗子

EXPLAIN select * from employees where name > 'a';

MySQL怎样选择合适的索引

如果用name索引查找数据需要遍历name字段联合索引树,然后根据遍历出来的主键值去主键索引树里再去查出最终数据,成本比全表扫描还高。

可以用覆盖索引优化,这样只需要遍历name字段的联合索引树就可以拿到所有的结果。

EXPLAIN select name,age,position from employees where name > 'a';

MySQL怎样选择合适的索引

可以看到通过select出的字段是覆盖索引,MySQL底层使用了索引优化。在看另一个case:

EXPLAIN select * from employees where name > 'zzz';

MySQL怎样选择合适的索引

对于上面的这两种 name>'a' 和 name>'zzz'的执行结果, mysql最终是否选择走索引或者一张表涉及多个索引, mysql最终如何选择索引,可以通过trace工具来一查究竟,开启trace工具会影响mysql性能,所以只能临时分析sql使用,用完之后需要立即关闭。

SET SESSION optimizer_trace="enabled=on",end_markers_in_json=on; --开启trace
SELECT * FROM employees WHERE name > 'a' ORDER BY position;
SELECT * FROM information_schema.OPTIMIZER_TRACE;

看trace字段:

{
 "steps": [
 {
  "join_preparation": { --第一阶段:SQl准备阶段
  "select#": 1,
  "steps": [
   {
   "expanded_query": " select `employees`.`id` AS `id`,`employees`.`name` AS `name`,`employees`.`age` AS `age`,`employees`.`position` AS `position`,`employees`.`hire_time` AS `hire_time` from `employees` where (`employees`.`name` > 'a') order by `employees`.`position`"
   }
  ] 
  } 
 },
 {
  "join_optimization": { --第二阶段:SQL优化阶段
  "select#": 1,
  "steps": [
   {
   "condition_processing": { --条件处理
    "condition": "WHERE",
    "original_condition": "(`employees`.`name` > 'a')",
    "steps": [
    {
     "transformation": "equality_propagation",
     "resulting_condition": "(`employees`.`name` > 'a')"
    },
    {
     "transformation": "constant_propagation",
     "resulting_condition": "(`employees`.`name` > 'a')"
    },
    {
     "transformation": "trivial_condition_removal",
     "resulting_condition": "(`employees`.`name` > 'a')"
    }
    ] 
   } 
   },
   {
   "table_dependencies": [ --表依赖详情
    {
    "table": "`employees`",
    "row_may_be_null": false,
    "map_bit": 0,
    "depends_on_map_bits": [
    ] 
    }
   ] 
   },
   {
   "ref_optimizer_key_uses": [
   ] 
   },
   {
   "rows_estimation": [ --预估标的访问成本
    {
    "table": "`employees`",
    "range_analysis": {
     "table_scan": { --全表扫描情况
     "rows": 3, --扫描行数
     "cost": 3.7 --查询成本
     } ,
     "potential_range_indices": [ --查询可能使用的索引
     {
      "index": "PRIMARY", --主键索引
      "usable": false,
      "cause": "not_applicable"
     },
     {
      "index": "idx_name_age_position", --辅助索引
      "usable": true,
      "key_parts": [
      "name",
      "age",
      "position",
      "id"
      ] 
     },
     {
      "index": "idx_age",
      "usable": false,
      "cause": "not_applicable"
     }
     ] ,
     "setup_range_conditions": [
     ] ,
     "group_index_range": {
     "chosen": false,
     "cause": "not_group_by_or_distinct"
     } ,
     "analyzing_range_alternatives": { ‐‐分析各个索引使用成本
     "range_scan_alternatives": [
      {
      "index": "idx_name_age_position",
      "ranges": [
       "a < name"
      ] ,
      "index_dives_for_eq_ranges": true,
      "rowid_ordered": false,
      "using_mrr": false,
      "index_only": false, ‐‐是否使用覆盖索引
      "rows": 3, --‐‐索引扫描行数
      "cost": 4.61, --索引使用成本
      "chosen": false, ‐‐是否选择该索引
      "cause": "cost"
      }
     ] ,
     "analyzing_roworder_intersect": {
      "usable": false,
      "cause": "too_few_roworder_scans"
     } 
     } 
    } 
    }
   ] 
   },
   {
   "considered_execution_plans": [
    {
    "plan_prefix": [
    ] ,
    "table": "`employees`",
    "best_access_path": {
     "considered_access_paths": [
     {
      "access_type": "scan",
      "rows": 3,
      "cost": 1.6,
      "chosen": true,
      "use_tmp_table": true
     }
     ] 
    } ,
    "cost_for_plan": 1.6,
    "rows_for_plan": 3,
    "sort_cost": 3,
    "new_cost_for_plan": 4.6,
    "chosen": true
    }
   ] 
   },
   {
   "attaching_conditions_to_tables": {
    "original_condition": "(`employees`.`name` > 'a')",
    "attached_conditions_computation": [
    ] ,
    "attached_conditions_summary": [
    {
     "table": "`employees`",
     "attached": "(`employees`.`name` > 'a')"
    }
    ] 
   } 
   },
   {
   "clause_processing": {
    "clause": "ORDER BY",
    "original_clause": "`employees`.`position`",
    "items": [
    {
     "item": "`employees`.`position`"
    }
    ] ,
    "resulting_clause_is_simple": true,
    "resulting_clause": "`employees`.`position`"
   } 
   },
   {
   "refine_plan": [
    {
    "table": "`employees`",
    "access_type": "table_scan"
    }
   ] 
   },
   {
   "reconsidering_access_paths_for_index_ordering": {
    "clause": "ORDER BY",
    "index_order_summary": {
    "table": "`employees`",
    "index_provides_order": false,
    "order_direction": "undefined",
    "index": "unknown",
    "plan_changed": false
    } 
   } 
   }
  ] 
  } 
 },
 {
  "join_execution": { --第三阶段:SQL执行阶段
  "select#": 1,
  "steps": [
   {
   "filesort_information": [
    {
    "direction": "asc",
    "table": "`employees`",
    "field": "position"
    }
   ] ,
   "filesort_priority_queue_optimization": {
    "usable": false,
    "cause": "not applicable (no LIMIT)"
   } ,
   "filesort_execution": [
   ] ,
   "filesort_summary": {
    "rows": 3,
    "examined_rows": 3,
    "number_of_tmp_files": 0,
    "sort_buffer_size": 200704,
    "sort_mode": "<sort_key, additional_fields>"
   } 
   }
  ] 
  } 
 }
 ] 
}

全表扫描的成本低于索引扫描, 索引MySQL最终会选择全表扫描。

SELECT * FROM employees WHERE name > 'zzz' ORDER BY position;
SELECT * FROM information_schema.OPTIMIZER_TRACE;

{
 "steps": [
 {
  "join_preparation": {
  "select#": 1,
  "steps": [
   {
   "expanded_query": " select `employees`.`id` AS `id`,`employees`.`name` AS `name`,`employees`.`age` AS `age`,`employees`.`position` AS `position`,`employees`.`hire_time` AS `hire_time` from `employees` where (`employees`.`name` > 'zzz') order by `employees`.`position`"
   }
  ] 
  } 
 },
 {
  "join_optimization": {
  "select#": 1,
  "steps": [
   {
   "condition_processing": {
    "condition": "WHERE",
    "original_condition": "(`employees`.`name` > 'zzz')",
    "steps": [
    {
     "transformation": "equality_propagation",
     "resulting_condition": "(`employees`.`name` > 'zzz')"
    },
    {
     "transformation": "constant_propagation",
     "resulting_condition": "(`employees`.`name` > 'zzz')"
    },
    {
     "transformation": "trivial_condition_removal",
     "resulting_condition": "(`employees`.`name` > 'zzz')"
    }
    ] 
   } 
   },
   {
   "table_dependencies": [
    {
    "table": "`employees`",
    "row_may_be_null": false,
    "map_bit": 0,
    "depends_on_map_bits": [
    ] 
    }
   ] 
   },
   {
   "ref_optimizer_key_uses": [
   ] 
   },
   {
   "rows_estimation": [
    {
    "table": "`employees`",
    "range_analysis": {
     "table_scan": {
     "rows": 3,
     "cost": 3.7
     } ,
     "potential_range_indices": [
     {
      "index": "PRIMARY",
      "usable": false,
      "cause": "not_applicable"
     },
     {
      "index": "idx_name_age_position",
      "usable": true,
      "key_parts": [
      "name",
      "age",
      "position",
      "id"
      ] 
     },
     {
      "index": "idx_age",
      "usable": false,
      "cause": "not_applicable"
     }
     ] ,
     "setup_range_conditions": [
     ] ,
     "group_index_range": {
     "chosen": false,
     "cause": "not_group_by_or_distinct"
     } ,
     "analyzing_range_alternatives": {
     "range_scan_alternatives": [
      {
      "index": "idx_name_age_position",
      "ranges": [
       "zzz < name"
      ] ,
      "index_dives_for_eq_ranges": true,
      "rowid_ordered": false,
      "using_mrr": false,
      "index_only": false,
      "rows": 1,
      "cost": 2.21,
      "chosen": true
      }
     ] ,
     "analyzing_roworder_intersect": {
      "usable": false,
      "cause": "too_few_roworder_scans"
     } 
     } ,
     "chosen_range_access_summary": {
     "range_access_plan": {
      "type": "range_scan",
      "index": "idx_name_age_position",
      "rows": 1,
      "ranges": [
      "zzz < name"
      ] 
     } ,
     "rows_for_plan": 1,
     "cost_for_plan": 2.21,
     "chosen": true
     } 
    } 
    }
   ] 
   },
   {
   "considered_execution_plans": [
    {
    "plan_prefix": [
    ] ,
    "table": "`employees`",
    "best_access_path": {
     "considered_access_paths": [
     {
      "access_type": "range",
      "rows": 1,
      "cost": 2.41,
      "chosen": true,
      "use_tmp_table": true
     }
     ] 
    } ,
    "cost_for_plan": 2.41,
    "rows_for_plan": 1,
    "sort_cost": 1,
    "new_cost_for_plan": 3.41,
    "chosen": true
    }
   ] 
   },
   {
   "attaching_conditions_to_tables": {
    "original_condition": "(`employees`.`name` > 'zzz')",
    "attached_conditions_computation": [
    ] ,
    "attached_conditions_summary": [
    {
     "table": "`employees`",
     "attached": "(`employees`.`name` > 'zzz')"
    }
    ] 
   } 
   },
   {
   "clause_processing": {
    "clause": "ORDER BY",
    "original_clause": "`employees`.`position`",
    "items": [
    {
     "item": "`employees`.`position`"
    }
    ] ,
    "resulting_clause_is_simple": true,
    "resulting_clause": "`employees`.`position`"
   } 
   },
   {
   "refine_plan": [
    {
    "table": "`employees`",
    "pushed_index_condition": "(`employees`.`name` > 'zzz')",
    "table_condition_attached": null,
    "access_type": "range"
    }
   ] 
   },
   {
   "reconsidering_access_paths_for_index_ordering": {
    "clause": "ORDER BY",
    "index_order_summary": {
    "table": "`employees`",
    "index_provides_order": false,
    "order_direction": "undefined",
    "index": "idx_name_age_position",
    "plan_changed": false
    } 
   } 
   }
  ] 
  } 
 },
 {
  "join_execution": {
  "select#": 1,
  "steps": [
   {
   "filesort_information": [
    {
    "direction": "asc",
    "table": "`employees`",
    "field": "position"
    }
   ] ,
   "filesort_priority_queue_optimization": {
    "usable": false,
    "cause": "not applicable (no LIMIT)"
   } ,
   "filesort_execution": [
   ] ,
   "filesort_summary": {
    "rows": 0,
    "examined_rows": 0,
    "number_of_tmp_files": 0,
    "sort_buffer_size": 200704,
    "sort_mode": "<sort_key, additional_fields>"
   } 
   }
  ] 
  } 
 }
 ] 
}

查看trace字段可知索引扫描的成本低于全表扫描的成本,所以MySQL最终选择索引扫描。

SET SESSION optimizer_trace="enabled=off"; -- 关闭tra

感谢你能够认真阅读完这篇文章,希望小编分享的“MySQL怎样选择合适的索引”这篇文章对大家有帮助,同时也希望大家多多支持亿速云,关注亿速云行业资讯频道,更多相关知识等着你来学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

MySQL怎样选择合适的索引

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

MySQL 选择合适的存储引擎

对于数据库这一块询问比较多的就是在 MySQL 中怎么去选择一种何时当前业务需求的存储引擎,而 MySQL 中支持的存储引擎又有很多种,那么 MySQL 中分别又有那些,怎么优雅的使用呢? 划分引擎原因 在文件系统中,MySQL 将每个数据
2022-05-24

MySQL 独立索引和联合索引的选择

通常会对多列索引缺乏理解,常见的错误是将很多列设置独立索引,或者是索引列使用错误的次序。我们在下一篇讨论索引列次序的问题,首先看一下多列独立索引的情况,以下面的表结构为例:CREATE TABLE test (c1 INT,c2 INT,c
2022-06-01

怎样选择适合自己的操作系统

如何选择适合自己的操作系统?选择适合自己需求的服务器操作系统是建立一个高效、安全和稳定的服务器环境的关键步骤之一。服务器操作系统的选择应该根据你的需求和要求来进行评估,以确保你能够获得好的性能和功能。以下是一些建议,帮助你选择适合自己需求的服务器操作系统。
怎样选择适合自己的操作系统
2024-02-01

怎么选择合适的VPS主机

选择合适的VPS主机需要考虑以下几个因素:配置需求:根据自己的网站或应用的需求,确定需要的CPU、内存、存储空间和带宽等配置。操作系统:选择适合自己应用的操作系统,如Linux、Windows等。数据中心位置:选择靠近目标用户或访问者的数据
怎么选择合适的VPS主机
2024-04-19

如何选择适合自己的MySQL版本?

如何选择适合自己的MySQL版本?在选择适合自己的MySQL版本时,我们需要考虑一些关键因素,如性能、安全性、功能特性等。不同的MySQL版本可能有不同的特性和优势,因此选择合适的版本对于系统的稳定性和性能至关重要。下面将介绍如何选择适合
如何选择适合自己的MySQL版本?
2024-03-15

MySQL为id选择合适的数据类型

目录分布式ID方案总结数据库自增ID数据库多主模式号段模式雪花算法选择 id 的数据类型,不仅仅需要考虑数据存储类型,还需要了解 MySQL 对该种类型如何计算和比较。例如,MySQL 将 ENUM 和 SET 类型在内部使用整型存储,但是
2022-05-16

怎样选择一种合适的网吧光纤路由器呢?

  路由器(Router),是连接因特网中各局域网、广域网的设备,它会根据信道的情况自动选择和设定路由,以最佳路径,按前后顺序发送信号。在这一篇教程里面,小编就会为大家简单的介绍一些怎样选择一种合适的网吧光纤路由器呢?希望这对大家有所帮助!  大家可以看到,网吧这一个行业作为当今社会最火红的行业,正在向着连锁化、规模化
怎样选择一种合适的网吧光纤路由器呢?
2024-04-18

xp电脑怎么选择合适的CPU

这篇文章主要介绍xp电脑怎么选择合适的CPU,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!1.成本一般来看,AMD的芯片要比英特尔同配置的便宜。不过英特尔酷睿i5处理器现如今有一个令人心动的价格,约为200美元。AM
2023-06-28

怎么选择适合的NoSQL数据库

选择适合的NoSQL数据库需要考虑以下几个因素:数据模型:根据应用程序的需求选择适合的数据模型,比如文档型、键值型、列族型或图形型。数据结构:根据数据的特性选择支持的数据结构,比如文档、键值、列族或图。数据量和性能:根据数据量和访问需求选择
怎么选择适合的NoSQL数据库
2024-05-07

编程热搜

目录