摘自:

http://www.educity.cn/wenda/590849.html

http://blog.csdn.net/hguisu/article/details/7106159

问:

不是说,一条sql语句只能用一个索引么

但SELECT * FROM `comment` WHERE `toconuid` = '10' or `tocomuid` = '10'

其中 toconuid列 和 tocomuid列 分别为单列索引

explain后 显示两个索引都用了,extra为 Using union(toconuid,tocomuid); Using where

答:

凡事总有特列。

而MYSQL可以理解为把这个语句拆成了两条语句SELECT * FROM `comment` WHERE `toconuid` = '10'unionSELECT * FROM `comment` WHERE `tocomuid` = '10'

在某些情况下,or条件可以避免全表扫描的。

1 .where 语句里面如果带有or条件, myisam表能用到索引, innodb不行。

1)myisam表:
 CREATE TABLE IF NOT EXISTS `a` (
  `id` int(1) NOT NULL AUTO_INCREMENT,
  `uid` int(11) NOT NULL,
  `aNum` char(20) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `uid` (`uid`)
) ENGINE=MyISAM  DEFAULT CHARSET=latin1 AUTO_INCREMENT=6 ;

mysql> explain select * from a where id=1 or uid =2;
+----+-------------+-------+-------------+---------------+-------------+---------+------+------+---------------------------------------+
| id | select_type | table | type        | possible_keys | key         | key_len | ref  | rows | Extra                                 |
+----+-------------+-------+-------------+---------------+-------------+---------+------+------+---------------------------------------+
|  1 | SIMPLE      | a     | index_merge | PRIMARY,uid   | PRIMARY,uid | 4,4     | NULL |    2 | Using union(PRIMARY,uid); Using where |
+----+-------------+-------+-------------+---------------+-------------+---------+------+------+---------------------------------------+
1 row in set (0.00 sec)

2)innodb表:

CREATE TABLE IF NOT EXISTS `a` (
  `id` int(1) NOT NULL AUTO_INCREMENT,
  `uid` int(11) NOT NULL,
  `aNum` char(20) DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `uid` (`uid`)
) ENGINE=InnoDB  DEFAULT CHARSET=latin1 AUTO_INCREMENT=6 ;

mysql>  explain select * from a where id=1 or uid =2;
+----+-------------+-------+------+---------------+------+---------+------+------+-------------+
| id | select_type | table | type | possible_keys | key  | key_len | ref  | rows | Extra       |
+----+-------------+-------+------+---------------+------+---------+------+------+-------------+
|  1 | SIMPLE      | a     | ALL  | PRIMARY,uid   | NULL | NULL    | NULL |    5 | Using where |
+----+-------------+-------+------+---------------+------+---------+------+------+-------------+
1 row in set (0.00 sec)

2 .必须所有的or条件都必须是独立索引:

+-------+----------------------------------------------------------------------------------------------------------------------
| Table | Create Table
+-------+----------------------------------------------------------------------------------------------------------------------
| a     | CREATE TABLE `a` (
  `id` int(1) NOT NULL AUTO_INCREMENT,
  `uid` int(11) NOT NULL,
  `aNum` char(20) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=MyISAM AUTO_INCREMENT=6 DEFAULT CHARSET=latin1 |
+-------+----------------------------------------------------------------------------------------------------------------------
1 row in set (0.00 sec)

explain查看:
mysql> explain select * from a where id=1 or uid =2;
+----+-------------+-------+------+---------------+------+---------+------+------+-------------+
| id | select_type | table | type | possible_keys | key  | key_len | ref  | rows | Extra       |
+----+-------------+-------+------+---------------+------+---------+------+------+-------------+
|  1 | SIMPLE      | a     | ALL  | PRIMARY       | NULL | NULL    | NULL |    5 | Using where |
+----+-------------+-------+------+---------------+------+---------+------+------+-------------+

1 row in set (0.00 sec)

全表扫描了。

3. 用UNION替换OR (适用于索引列)

通常情况下, 用UNION替换WHERE子句中的OR将会起到较好的效果. 对索引列使用OR将造成全表扫描.

注意, 以上规则只针对多个索引列有效. 如果有column没有被索引, 查询效率可能会因为你没有选择OR而降低.

在下面的例子中, LOC_ID 和REGION上都建有索引.
       高效:

  1. select loc_id , loc_desc , region from location where loc_id = 10
  2. union
  3. select loc_id , loc_desc , region  from location where region = "melbourne"

低效:

  1. select loc_id , loc desc , region from location where loc_id = 10 or region = "melbourne"

如果你坚持要用OR, 那就需要返回记录最少的索引列写在最前面.

4. 用in来替换or

这是一条简单易记的规则,但是实际的执行效果还须检验,在oracle8i下,两者的执行路径似乎是相同的. 
低效: 
select…. from location where loc_id = 10 or loc_id = 20 or loc_id = 30 
高效 
select… from location where loc_in  in (10,20,30);

来源:mysql关于or的索引问题 - 爱搁浅 - 博客园

mysql关于or的索引问题相关推荐

  1. Mysql InnoDB B+树索引和哈希索引的区别? MongoDB 为什么使用B-树?

    B-树和B+树最重要的一个区别就是B+树只有叶节点存放数据,其余节点用来索引,而B-树是每个索引节点都会有Data域. B+树 B+树是为磁盘及其他存储辅助设备而设计一种平衡查找树(不是二叉树).B+ ...

  2. mysql查看使用的索引_Mysql查看是否使用到索引

    mysql数据库创建索引优化之后,在查询时想看下是否使用到索引, 使用执行计划查看: mysql> explain  SELECT * FROM tb_user WHERE STATUS=1 l ...

  3. psql where里有自定义函数慢_阿里P8架构师谈:MySQL慢查询优化、索引优化、以及表等优化总结...

    MySQL优化概述 MySQL数据库常见的两个瓶颈是:CPU和I/O的瓶颈. CPU在饱和的时候一般发生在数据装入内存或从磁盘上读取数据时候. 磁盘I/O瓶颈发生在装入数据远大于内存容量的时候,如果应 ...

  4. MySQL force Index 强制索引概述

    以下的文章主要介绍的是MySQL force Index  强制索引,以及其他的强制操作,其优先操作的具体操作步骤如下:我们以MySQL中常用的hint来进行详细的解析,如果你是经常使用Oracle的 ...

  5. Oracle与Mysql主键、索引及分页的区别小结

    Oracle与Mysql主键.索引及分页的区别,学习oracle的朋友可以参考下 区别: 1.主键,Oracle不可以实现自增,mysql可以实现自增. oracle新建序列,SEQ_USER_Id. ...

  6. Mysql如何创建短索引(前缀索引)

    Mysql如何创建短索引 为什么要用短索引 有时需要索引很长的字符列,它会使索引变大并且变慢.一个策略就是模拟哈希索引.但是有时这也不够好,那么应该怎么办呢? 通常可以索引开始的几个字符,而不是全部值 ...

  7. mysql 大量数据 更改索引_Mysql索引数据结构详解与索引优化

    本篇文章主要学习了MySQL的索引的数据结构的认识,做一个大概的了解即可. 一.索引 在关系数据库中,索引是一种单独的.物理的对数据库表中一列或多列的值进行排序的一种存储数据结构,它是某个表中一列或若 ...

  8. MySQL中的联合索引学习教程

    MySQL中的联合索引学习教程 这篇文章主要介绍了MySQL中的联合索引学习教程,其中谈到了联合索引对排序的优化等知识点,需要的朋友可以参考下 联合索引又叫复合索引.对于复合索引:Mysql从左到右的 ...

  9. mysql调试索引_10 分钟让你明白 MySQL 是如何利用索引的?

    一.前言 在 MySQL 中进行 SQL 优化的时候,经常会在一些情况下,对 MySQL 能否利用索引有一些迷惑. 譬如: MySQL 在遇到范围查询条件的时候就停止匹配了,那么到底是哪些范围条件? ...

  10. 面试mysql中怎么创建索引_阿里面试:MySQL如何设计索引更高效?

    有情怀,有干货,微信搜索[三太子敖丙]关注这个不一样的程序员. 本文 GitHub https://github.com/JavaFamily 已收录,有一线大厂面试完整考点.资料以及我的系列文章. ...

最新文章

  1. NC:蜜蜂肠道微生物的多样性
  2. 20165315 第八周考试课下补做
  3. 系统架构图怎么画_简历的项目经历应该怎么写得好看?
  4. 使用直线标定板进行相机畸变校正,并且进行9点标定(halcon)
  5. 刘强东为抗疫发声:我们送的不是货,是温暖和希望!
  6. linux get current thread count and system threads limit
  7. matlab中用数据拟合圆心,拟合圆并求圆心(matlab)
  8. kettle复杂表头的Excel数据源处理
  9. Qualcomm MPM introduction
  10. 隐藏百度地图logo
  11. “ODM OEM OBM的区别”网址汇总
  12. 高中数学数列技巧解题秒杀视频:数列小题秒杀技巧
  13. HACCP原理——确定关键控制点(转载)
  14. 某酷ckey签名生成算法系列--(三)ast代码控制流平坦化
  15. linux的nfs端口111,【NFS】NFS设置固定端口,添加防火墙规则
  16. SVA 断言翻译笔记 16.13多时钟序列语法(九)
  17. 如何搜索的时候去除电驴emule的乱码显示
  18. 【PWN系列】格式化字符串在bss段上的处理
  19. html 页面滚动时 div位置不变,js实现页面刷新滚动条位置不变
  20. fx5800p编程教程_常见的CASIO fx5800P基本功能及编程学习.ppt

热门文章

  1. QT的QDataStream类的使用
  2. QML基础类型之bool
  3. pandas.DataFrame.iloc的使用
  4. ElasticSearch技术文档
  5. 第29天:控制进度,控制进度知识点,敏捷中控制进度知识点
  6. AdventureWorksDW 2019还原图解
  7. 中如何使用echart_jQueryEasyUI中的拖拽事件如何使用
  8. 计算机控制试卷中南大学,期末试卷,需要的自取
  9. linux自动挂起什么意思,Linux中进行挂起(待机)的命令说明
  10. 获取input内容并回填_超详细的软件测试内容实战