mysql的sql语句优化5种方式

mysql的sql语句优化5种方式,第1张

只有5种吗?我知道十种以上的说。

索引(没我得全表查询了)

改变数据储引擎(MyISAM没事务再也不用担心锁表了)

增加冗余数来减少连表查询数(消耗硬盘空间减少CPU使用)

调整查询顺序减少查询量优先(数量少了连表的笛卡儿积也少了)

全文索引(文字长度有限制,而且IO使用量会大增,但是妥妥的快)

查询尽量不要用函数(函数可是不走索引的哦亲)

查询变量类型要提前对好减少系统负担(我提前改变了系统你就不用检测了)

升级服务器硬件(没什么是氪金解决不了的)

配置好临时表空间,合理理由临时表减少主表查询抢资源(唯我独查)

合理理由函数减少系统的判断(明明都能确认内容不同你用UNION 系统还是傻傻的查一遍是否重复 UNION  ALL则跳过这个步骤同理 inner join 和 left join 也一样 )

强制走索引(复合索引的情况有时候手动走比系统判断要好哦)

脏读、幻读等(你堵车我绕路)

数据归档,迁移(没用的数据要进仓哦,别占着主表的资源)

表的碎片整理(迁移后碎片整理更健康哦亲)

索引重构(数据都走了索引也应该重构一下才能保证速度哦)

善用存储过程(串N个表(N大于10)的查询千万别一个SQL到底,分布式查询在吧结果集合并吧骚年)

预处理数据(mysql也有job哦,对于经常要子查询的数据可以先弄个明细表根据主表在后台进行补完,查询的时候就更方便了)

懒得说了。。。。。。。。。。。。。。。。。。

mysql优化是一个大方向,大的是要分布式、读写分离,小的是对sql语句进行优化。不过大多问的也是对sql语句优化,网上很多资料,我就大体说说。

1、explain+索引。

在你要查询的语句前加explain,看下有没有用到索引,如果出现type为all的,则说明有必要添加下索引。(附多表查询速度比较:表关联>exists>in)慢查询优化是一大块。

2、预统计。

很经常需要对历史的数据进行过滤统计。比如移动需要统计上个月电话小时数超过N小时的人,那么如果直接取原始数据,那将很慢,此时如果每天晚上凌晨都对数据进行预统计,统计每个人每天电话时数,那再来过滤就很快。

3、分表分区。

分表分区也是为了提高搜索速度。例如,公交车的gps行驶记录,gps每隔15s报一次,一辆车一天运行12小时,一天就要插入4*60*12条记录,N辆车就要再乘,其数量极大,所以经常按月分表,分表里再按上报时间做日分区,这样就达到很大的优化,想查询某段时间,mysql很快就可以定位到。

4、表结构。

表结构很重要,经常需要多表关联查询一些字段,有时可以冗余下放到同一张表。

mysql优化很有意思,多去查阅些资料,多去尝试,对你有好处的。

创建表tb_point 表

准备空的tb_box表

函数

编写存储过程,给tb_box表添加100万条数据

修改关联数据

好于

优于

在执行以下语句时会报错:

前面在 https://www.jianshu.com/p/95e50fd017ea 文章中有提到这个问题,是直接修改sql_mode将 ONLY_FULL_GROUP_BY直接干掉。但是在《高性能mysql》中有一段话是这样的:

那么既然指出不要直接修改 sql_mode,那么我们应该如何让冲突的GRUOPBY语句正确执行呢?

文中有提到,可以使用max()和min()函数来实现;但是这种方式使用max和min函数较真的人可能会说这样写的分组查询有问题,确实如此。但是如果更加在乎查询效率,这样做也无可厚非。

如果,实在无法接受使用上面那种方式的话,可以这样使用子查询的方式来进行查询:

书上对于这种方式有描述如下:

这样写更满足关系理论,但是成本有点高,因为子查询需要填充临时表,而子查询中创建的临时表是没有任何索引的。

作者认为这样写对性能有影响。

但是从我测得结果来看,子查询的耗时反而更少。性能反而更佳。这个子查询耗时0.4秒。而使用max方式耗时0.8秒。几乎一倍。我的mysql版本是 5.7.22-log

为了解其中的原因,我们查看它的执行计划:

可见,因为子查询而产生了一层 DERIVED 临时表,但是这个临时表的Extra字段有显示 Using index、key里面显示自建索引。说明用到了索引。这是查询性能可观的一个重要原因吧;

另外我分别使用 SHOW PROFILE命令查看各部分耗时,对比之下。没看到有哪部分耗时差别特别大,使用JOIN、MAX 耗时比上子查询耗时都差不多是1倍

有些时候对一没有建立索引的字段,进行GRUOP BY时。会产生Using filesort 文件内排序。因为GRUOP BY是在排序的基础上进行分组的。

如下面sql:

如果业务上不对排序有要求。那么就可以禁止GRUOP BY的排序:

这样就把Using filesort给干掉了! 执行时间 1.237

当然,多数情况是多排序有要求的。此时也可以在GRUOP BY后面使用DESC和ASC关键字,使分组的结果集按需要的方向排序。如下:

分组查询的一个变种就是要求mysql对分组结果再进行一次超级聚合。可以使用GROUP BY WITH ROLLUP 来实现这种逻辑,但可能性能不佳。因为通过查询计划分析出它是使用 Using temporaryUsing filesort 来实现的。

使用WITH ROLLUP,查询时间2.531秒。不使用0.774 秒。

1、所以,很多时候。我们在应用程序中做超级聚合是最好的!

2、当然也可使用UNION ALL 来实现:

3、还可以通过FROM子句嵌套使用子查询:


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/zaji/7325197.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-04-04
下一篇2023-04-04

发表评论

登录后才能评论

评论列表(0条)

    保存