mssql 计划怎每隔n秒_详解一条数据库sql优化案例--从23秒到0.9秒

概述

新项目业务人员反馈说最近订单发放模块经常很卡,导致总是有锁的情况发生,在用慢查询和开启锁监控观察后发现实际上只是单条查询慢造成的阻塞锁,这里考虑先对单条查询做一下优化。


1、优化前的表结构、数据量、SQL、执行计划、执行时间

1.1、表结构

A表有90个字段,B表有140个字段。

c338ace870e130ddb00d29165caa50f8.png

1.2、数据量

select count(*) from A;--166713select count(*) from B;--220810

1.3、sql

开启慢查询观察到慢sql如下,单条执行只取200条记录是在23秒左右。

select ob.id, ob.customer, ob.order_no1, ob.accountingitems_code, ob.insert_date, ob.weight, ob.volume, ob.qty, ob.project_code,ob.order_no2,ob.order_type1  from A as ob where ifnull(ob.project_code,'')<>'' and ifnull(ob.accountingitems_code,'')<>'' and ob.if_cost_proof='N' and EXISTS (select 1 from B ol where ob.id=ol.order_base) limit 200;
c7a30f62a8ec3e6d42f9f66ce93f09bc.png
6673b458d969ea80148f0ada7447cdc2.png

1.4、执行计划

2d82c56b80f6f8a7c07f5337d35cf544.png

思路

这两张表都是订单表,全国各地的每天大概会产生十万行左右,这里又是全扫,等后期达到上千万的数据就GG了。目前只是看到这个sql上的问题,先考虑exists部分做一下改写。


2、exists部分改写

select ob.id, ob.customer, ob.order_no1, ob.accountingitems_code, ob.insert_date, ob.weight, ob.volume, ob.qty, ob.project_code,ob.order_no2,ob.order_type1  from fsl_order_base as ob,fsl_order_base_line olwhere ob.id=ol.order_base and ob.if_cost_proof='N' andifnull(ob.project_code,'')<>'' and ifnull(ob.accountingitems_code,'')<>'' limit 200;

执行时间:耗时1.8秒

e4de1016d009a65531058631ac400482.png

对应的执行计划:

可以看到ob表走了主键索引

e0a913feaa38b2d21413edbc3d75f9ec.png

业务确认结果符合需求,那就在这基础上建一下索引吧!


3、ol表建索引

create index idx_obl_id on fsl_order_base_line(order_base);create index idx_ob_id_cost on fsl_order_base(id,if_cost_proof); --加上去但实际上用不到这个索引,选择去掉
e0c84b56e6e9a6444101226c022ecb66.png

4、查看执行时间和执行计划

耗时1.1秒,可惜执行计划还是走了全扫,在对ob表建了索引实际上也用不到,最终只在ol表建了索引。

f899f59a6b4adfb6eefe3035f38cecd3.png
07a48c9b23e60fe61ae2ecf6f91bbd0f.png

5、考虑用join改写

把ob结果集缩小,然后再做关联查,并测试是否可以用上索引。

SELECT obc.id, obc.customer, obc.order_no1, obc.accountingitems_code, obc.insert_date, obc.weight, obc.volume, obc.qty, obc.project_code, obc.order_no2, obc.order_type1 FROM (select * from fsl_order_base AS ob where ob.if_cost_proof = 'N' and ifnull( ob.project_code, '' ) <> '' and ifnull( ob.accountingitems_code, '' ) <> '' ) obc join fsl_order_base_line ol  on obc.id = ol.order_base limit 200;

时间快了一点,但不是很明显,先凑合吧

c2bae30708287a331bf8491fe51ee5ea.png

执行计划保持不变。

6fdff20c26a6a445dbb4ba290b10b2a7.png

总结

建索引前因为走了主键索引,所以时间在1.6秒这样,建索引后不走主键索引了,走ol表的索引,所以在1.5秒,然后缩小结果集去查的话就在1s这样。

更重要的是这两个表一个90个字段,一个150个字段,所以这两个表的关联查后期结果集应该还是会很大,建议是弄成分区表的形式,表能拆分的话是最好的。这些长度不要直接给那么大,这么宽对性能都是有影响的。

7ef6aaac74339caef97f570b7df4228b.png

觉得有用的朋友多帮忙转发哦!后面会分享更多devops和DBA方面的内容,感兴趣的朋友可以关注下~

23450a7190c91d869067c1ea122a3112.gif
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值