sqlserver中几种典型的等待

 更新时间:2016年05月28日 21:06:14   作者:hidba  
在最近的几次sqlserver问题的排查中,总结了sqlserver几种典型的等待类型,类似于oracle中的等待事件,如果看到这样的等待类型时候能够迅速定位问题的根源,下面通过一则案例来把这些典型的等待处理方法整理出来

为了准备今年的双11很久没有更新blog,在最近的几次sqlserver问题的排查中,总结了sqlserver几种典型的等待类型,类似于oracle中的等待事件,如果看到这样的等待类型时候能够迅速定位问题的根源,下面通过一则案例来把这些典型的等待处理方法整理出来:

第一种等待.memory等待

早上接到一用户反馈其RDS实例非常的慢,通过观察sqlserver活动会话监视器(active monitor)的waiting tasks(类似于mysql的thread running)可以看到有10多w的等待任务,可以明确数据库现在已经出现了较大的瓶颈,紧接着通过resource waits看到数据库中有大量的memory内存等待:

看到是memory 资源等待后,为了立刻恢复用户应用,想到立刻去调大内存,发现该实例已经是24G了,看来一下os的空余内存,还有较多的内存剩余,所以将内存调大到36G,发现resource waits还是在memory上等待,同时这个时候的cpu使用率飙升,达到了90%左右(之前在10%左右的等待).这样解决不了根本问题,于是通过recent expensive queries,发现以下sql的逻辑读很高,执行非常频繁:

SELECT * FROM RefundOrder_Message messages0_ WHERE messages0_.Order_Id=@p0;

也可以通过如下方式获得造成内存等待的sql:
SELECT st.text FROM sys.dm_exec_query_memory_grants req CROSS APPLY sys.dm_exec_sql_text(req.sql_handle) as ST where req.grant_time is NULL or req.granted_memory_kb is NULL

The columns grant_time and granted_memory_kb will be NULL for those queries which are waiting to get their requested memory

sp_helpindex RefundOrder_Message
发现该表只有一个主键索引:

创建一下索引:
create index ind_RefundOrder_Message_order_id on RefundOrder_Message(Order_Id);

第二种等待:latch等待


在索引加上去后,memory的等待立刻消失,但是resource waits的等待变为了 lock:

通过以下内部视图可以发现如下调用出现了等待:
SELECT ss.host_name, req.blocking_session_id,req.wait_type ,req.wait_time ,req.wait_resource ,req.transaction_id ,st.text FROM sys.dm_exec_requests req CROSS APPLY sys.dm_exec_sql_text(req.sql_handle) as ST
cross apply sys.dm_exec_sessions ss where req.status =N'suspended' and ss.session_id=req.session_id;

得到阻塞其他会话的sql:
(@p0 int,@p1 nvarchar(4000),@p2 bit)
SELECT TOP (@p0) this.* FROM ViewSalesOrder this_ WHERE this_.MemberCode = @p1 and this_.IsObsolete = @p2 ORDER BY this_.OdCode desc;

视图ViewSalesOrder是一张非常核心的视图,里面关联了订单,订单消息,订单发货等多个业务逻辑;查询条件中代入了membercode为店铺的名称,可能操作某个店铺的订单;
通过ViewSalesOrder视图中的定义,membercode,IsObsolete ,OdCode 为salesOrder表的三个字段,查看salesOrder上并没有相应的索引,于是加上如下索引:
create index ind_salesOrder_member on salesOrder(membercode,IsObsolete,code);
在添加完索引后,数据库的waiting tasks 下降,batch requests提升:

第三种等待:lock

第三种等待是常见的等待,常见的情况在删除,更新的时候由于条件中没有合适的索引导致锁定的记录范围太大,导致阻塞其他的会话请求:

用户在在进行压测的时候发现一条更新语句执行的非常慢,导致整个系统都卡住:

update DD_ShenHe   set ZF = 0   where zf is null;

查看dd_shenhe表上面的索引:

可以看到表中并没有zf字段的索引,而该表总共有400w的数据,zf 为null的有8000条,所以在zf字段添加索引是合适的:

Create index ind_dd_shenhe_zf on dd_shenhe(zf);

添加完索引后,系统恢复正常。

相关文章

  • SQL Server数据库表的创建与管理操作大全

    SQL Server数据库表的创建与管理操作大全

    这篇文章主要给大家介绍了关于SQL Server数据库表的创建与管理操作的相关资料, SQL Server是一个关系型数据库管理系统,它可以帮助用户创建、管理和查询数据,文中介绍的非常详细,需要的朋友可以参考下
    2024-01-01
  • 总结一周内学习的Sql经验(一)

    总结一周内学习的Sql经验(一)

    学习sql有一周时间了,这些主要是基础的内容,以后估计就遇到具体的实例再拿出来分析。
    2014-08-08
  • sql字符串函数大全和使用方法示例

    sql字符串函数大全和使用方法示例

    本文主要讲解SQL SERVER的字符串函数使用方法,包括LEN、LOWER、UPPER (string)、LTRIM、RTRIM等函数
    2014-01-01
  • SqlServer2000+ 身份证合法校验函数的示例代码

    SqlServer2000+ 身份证合法校验函数的示例代码

    这篇文章主要介绍了SqlServer2000+ 身份证合法校验函数,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • SQL Server数据库生成与执行SQL脚本详细教程

    SQL Server数据库生成与执行SQL脚本详细教程

    为了方便可以把需要连续执行的SQL语句写到一个文本文件中,并且用.SQL作为扩展名,这种文件叫做SQL脚本文件,下面这篇文章主要给大家介绍了关于SQL Server数据库生成与执行SQL脚本的相关资料,需要的朋友可以参考下
    2023-01-01
  • SQL Server 2016 无域群集配置 AlwaysON 可用性组图文教程

    SQL Server 2016 无域群集配置 AlwaysON 可用性组图文教程

    这篇文章主要介绍了SQL Server 2016 无域群集配置 AlwaysON 可用性组图文教程,需要的朋友可以参考下
    2017-04-04
  • SQLServer 清理日志的实现

    SQLServer 清理日志的实现

    本文主要介绍了SQLServer 清理日志的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-01-01
  • Sql检验数据库中各个表的数据量

    Sql检验数据库中各个表的数据量

    本文主要介绍了Sql检验数据库中各个表的数据量,这样的查询有助于评估每个表的数据量,从而更好地进行数据库性能优化,具有一定的参考价值,感兴趣的可以了解一下
    2023-12-12
  • 完美解决SQL server2005中插入汉字变成问号的问题

    完美解决SQL server2005中插入汉字变成问号的问题

    以下是对在SQL server2005中插入汉字变成问号的解决办法进行了分析介绍。需要的朋友可以过来参考下
    2013-08-08
  • SQL之CASE WHEN用法小结

    SQL之CASE WHEN用法小结

    CASE WHEN语句通常用于根据条件执行不同的操作,本文主要介绍了SQL之CASE WHEN用法小结,具有一定的参考价值,感兴趣的可以了解一下
    2024-05-05

最新评论