SQLServer2012 分页语句执行分析
上一篇文章提到了,SQLServer2012在使用Offset,Fetch语句分页时,获取了大量不需要的数据,导致查询效率低的问题。
现在让我们来看看,究竟是什么导致SQLServer不能按需取数呢?
1. 先来看看语句,及其执行计划
SELECT BillDate, SUM([QTY]) PosInQtyFROM RtlDetail A,RtlBill B WHERE A.billno = B.billno AND BillDate>='2009-09-25 00:00:00' AND BillDate<='2012-09-25 23:59:59'GROUP BY BillDateORDER BY BillDateOFFSET 10 ROW FETCH NEXT20 Rows Only;分析至此,认为并行计算在这里拖后腿了。如果没有并行计算,由于在排序列上是有索引的,取数都是按排序的规则进行,那么一旦取得了需要的结果子集,处理就可以结束了。也就可以避免IO及CPU的浪费。
3. 验证一下想法,强制设置最大并行度为1,再执行语句
SELECT BillDate, SUM([QTY]) PosInQty
FROM RtlDetail A,RtlBill B
WHERE A.billno = B.billno AND BillDate>='2009-09-25 00:00:00' AND BillDate<='2012-09-25 23:59:59'
GROUP BY BillDate
ORDER BY BillDate
OFFSET 10 ROW FETCH NEXT20 Rows Only option (maxdop 1);
耗时: 1s (并行时18s秒啊,坑爹啊,有木有。。。)
好的,这个效率终于能和window下的MariaDB媲美了,可人家在Linux下执行只需要 265ms 啊。 呵呵,这个是后话了。
4. 建议
(1)对微软的建议:在生成分页语句的执行计划时,可以先比较下并行的成本和非并行的成本,如果差距太大,就不要用parallel的方式了;
(2)对使用者的建议:如果你的分页语句耗时太长,又常常只获取前面的部分分页内容,那么就把并行关了吧。至少在微软做出优化之前。