SQL性能优化之分页
讨论一下无主键分页问题。
还是以实际业务场景为例:
后台有一张存货编码Inventory,注意该表为基础档案表,涉及的控制字段极多,且由于历史原因设置的料品编码为主键
首先查询一下数据量
数据量才不到两万,但是分页查询请求已经超时,后台抓一下sql语句,如下:
select * from (select row_number() over(order by pubufts desc) n, * from (select a.cInvCode,a.cInvName,a.cInvStd,a.cComUnitCode,a.bExpSale,a.pubufts,a.bInvQuality,a.bInvBatch,a.iMassDate,a.bPurchase,a.bSelf,a.bComsume,a.bProducing,a.bSale,a.bBomSub,b.cComUnitName,BInv.cLotPOVType,iValidDays
FROM dbo.Inventory a
left join ComputationUnit b on b.cComunitCode=a.cComUnitCode
left join KL_InvLotParam as BInv on a.cInvLotParamCode = BInv.cInvLotParamCode) as t ) as t where n >= 1 and n <= 30
反复执行后依然需要十多秒的时间,接下来看一下执行计划:
发现主要开销在于排序Sort和整表检索Scan,再来看看sql语句,发现使用row_number使用pubufts排序,这个字段是时间戳字段,格式如下:0x0000000004450C91
再试试把排序换成有索引的主键cInvCode,查询速度不到一秒。
再次使用新的方法测试,sql如下:
SELECT * FROM Inventory
OFFSET子句:用于指定跳过的数据行;
FETCH子句:该子句在OFFSET子句之后执行,表示在跳过指定数量的数据行之后,返回一定数据量的数据行;
执行顺序:OFFSET子句必须在Order By 子句之后执行,FETCH子句必须在OFFSET子句之后执行;
虽然只是一个索引导致的简单问题,但是确实没想到效率会差这么多。
以后有更好的方案再继续补充吧
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?
· Pantheons:用 TypeScript 打造主流大模型对话的一站式集成库