事情的过程是:公司有一个上百行的sql 运行在MySQL数据库,速度奇慢无比,逻辑乱七八糟,我就不贴出来了,经过这次修改想总结一下如何写一个不被人骂的sql。

说一些被人诟病的问题:

一、子查询

  把你的子查询全部改为join!!! 把你的子查询全部改为join!!! 把你的子查询全部改为join!!!  不要搞子查询,因为子查询的速度真的很慢。

把你用到的表先准备出来,找好关联关系做成 left join 或者 inner join 别问我为什么不用 right join mysql里面没有啊~

二、没索引

  几个必须加索引的地方:

  1.主键自动建立唯一索引

  2. 频繁作为where条件语句查询的字段

  3. 关联字段需要建立索引,比如join表 on的字段

  4. 排序字段可以建立索引

  5. 分组字段可以建立索引,因为分组的前提是排序

  6. 统计字段,聚合函数 可以建立索引,例如count(),max()

  当然索引也不是哪都合适的:

  1.频繁更新的字段不适合建立索引

  2.where条件中用不到的字段不适合建立索引

  3.表数据可以确定比较少的不需要建索引 少于1000条

  4.数据重复且发布比较均匀的的字段不适合建索引(唯一性太差的字段不适合建立索引),例如性别,状态

  5. 参与列计算的列不适合建索引

  索引什么时候会失效:

  1. 索引中不能有列的值是null 所以 is null 或者 not null 索引不起作用

  2. 单独引用复合索引里非第一位置的索引列:比如索引是a,b,c  那么 a / a,b / a,b,c 都可以使用索引,而 b / b,c 则不行

  3. 对索引列运算 运算包括(+、-、*、/、!、<>、%、like’%_’(%放在前面)、or、in、exist等),导致索引失效。

  4. 对索引应用内部函数 比如: select * from table_A  where ROUND(id) = 1 此时应该建ROUND(id)为索引。

  5. 类型错误,如字段类型为varchar,where条件用number。

例:id字段是varchar类型。

错误写法:select * from table where id= 1

正确写法:select * from table where id = ‘1’

  6. 当or 条件时 必须所有的条件都是独立索引才能走索引

  7. mysql查询只能使用一个索引,如果where中已经使用了索引,那么order by 中就无法使用。前半句举例:select * from table where xxx= '1' and yyy =2 如果 xxx,yyy都时独立索引 那么只能xxx走了索引,其实已经很快了,如果想更快就建组合索引。

  创建索引

  1. 普通索引:create index '索引名'  on '表名' ('字段名‘(length));  alter table '表名' add index '索引名' ('字段名 (ength));

  2. 唯一索引 可以为空 :create unique index '索引名'  on '表名' (字段名(length));

  3. 主键索引 唯一切不为空:alter table ‘表名’ add primary key(‘列名’);

  4. 组合索引:create index '索引名'  on '表名'('字段名1’,'字段名2’);

三、正确选择 in , exists 和inner join

  1. 如果集合比较小的时候,选择in 因为 in 先查询子集

  2. 如果集合比较大的时候,选择exists 因为 exists 先查询主查询 exists 简单使用:

  SELECT c.CustomerId,CompanyName FROM Customers c WHERE EXISTS( SELECT OrderID FROM Orders o WHERE o.CustomerID=c.CustomerID)

  这里面的EXISTS是如何运作呢?子查询返回的是OrderId字段,可是外面的查询要找的是CustomerID和CompanyName字段,这两个字段肯定不在OrderID里面啊,这是如何匹配的呢?

  EXISTS用于检查子查询是否至少会返回一行数据,该子查询实际上并不返回任何数据,而是返回值True或False 。

  EXISTS 指定一个子查询,检测 行 的存在。

  语法: EXISTS subquery

  参数: subquery 是一个受限的 SELECT 语句 (不允许有 COMPUTE 子句和 INTO 关键字)。

  结果类型: Boolean 如果子查询包含行,则返回 TRUE ,否则返回 FLASE 。

  3. 如果是not 就选exists 吧 。

  4. inner joiin 如果你用 in 或者 exists 是会返回主表的行  但是inner join 如果子查询有重复值的话 就会返回重复行 。

最新文章

  1. php 2038年问题
  2. SVM-线性可分支持向量机
  3. ACM Binary String Match
  4. javascript中字符串常用方法总结
  5. 查找一个数组中最小的前n项
  6. Java系统属性与Preferences API的简单介绍
  7. eclipse导入项目之后有感叹号
  8. 小甲鱼OD学习第12讲
  9. Gradle 1.12用户指南翻译——第二十三章. Java 插件
  10. python 字符串转化为json、post请求
  11. 对Datatable中过长内容实行省略话
  12. HTML5 学习05—— 拖放(Drag 和 Drop)
  13. drf 认证、权限、限流、过滤、排序、分页器
  14. Python Django 之 简单入门
  15. asp.net MVC 导出excle(转)
  16. Docker run centos 内部使用systemctl 启动服务的方法
  17. How could I create a custom windows message?
  18. idong常用js总结
  19. Java程序员的日常—— Spring Boot单元测试
  20. 使用C#的两种方式OracleClient组件和OleDB组件连接ORACLE数据库

热门文章

  1. 基于SpringBoot + Mybatis实现 MVC 项目
  2. linux下mysql数据库操作命令
  3. .Net Core 微服务容器系列基础目录篇
  4. JavaScript 面向对象编程 &#183; 理解对象
  5. robotframework框架 - seleniumLibrary 关键字解读-全攻略
  6. SpringBoot-ElasticJob封装快速上手使用(分布式定时器)
  7. 什么是线程调度器(Thread Scheduler)和时间分片(Time Slicing )?
  8. [以太坊源代码分析] I.区块和交易,合约和虚拟机
  9. openpyxl中遇到TypeError: &#39;generator&#39; object is not subscriptable的问题和解决方案
  10. Dubbo和Zookerper的关系