概述:
当一个SQLServer实例运行得很慢的时候,应该做一些检查,如检查等待状态。最好的方法是一开始就建立一个性能基线,以便做性能对比。当发现与性能基线对比后,存在内存压力的话,就要找出是什么原因导致的。可以检查事务的等待状态,其中Resource_semaphore等待可能出现最多。下面是如何去处理这个问题:
当检查事务的所有等待类型后,可能会发现Resource_semaphore这个等待类型出现非常多,这会增加一些页面的IO等待。因为这些事务没有足够的内存来处理它们的操作,所以到这了页面的IO等待。
Resource_semaphore等待:
首先我们来弄清楚一下什么是Resource_semaphore等待。当SQLServer收到一个用户请求(或者查询时)。首先会创建一个编译后的计划,然后在这个基础上创建一个执行计划。当SQLServer创建一个编译后的计划时,它会计算两个内存授予参数,成为:请求内存(required memory)和额外内存(additional memory)
请求内存是运行排序和hash连接的所需最少内存,之所谓成为“请求”,是因为查询不需要在一开始就申请这部分的内存。而额外内存是存放临时数据到内存中所需的那部分内存。如果没有足够的内存,查询所需的数据将会存到硬盘当中。
首先,服务器会计算运行特定查询所需要的内存。这部分通常等于请求内存和额外内存的总和。但当实例使用并行执行时,所需的内存为(请求内存*并行度)加上额外内存的总和。服务器会检查是否有足够的内存来运行每个查询,然后会降低额外内存的量,知道所有总内存需求量刚好达到内存的限制量。这部分修改后的内存成为需求内存(requested memory)。在SQLServer内不能,有一个叫Resource Semaphore的设置,用于授予需求内存用于查询。当查询没有得到足够的内存,就会把等待状态改为:Resource_Semaphore。可以从sysprocesses系统表或者sys.dm_exec_request
DMV中查询。
当Resource_semaphore接受一个新的请求时,首先检查是否有查询还在等待中,如果发现有,那么会把这个新请求放到先进先出的队列中,Resource Semaphore会尝试对未等待的查询授予内存,这部分内存可能是之前的查询执行完毕后返回的内存。如果发现有足够的内存,那么就会把内存赋予给处于Resource Semaphore等待状态的查询,让其开始运行。如果不够,那么会把查询放入等待队列并标记为Resource_Semaphore等待。因此,看这个等待状态可以发现内存存在压力。
识别Resource_Semaphore等待:
步骤1:
执行以下语句,并筛选Resource_Semaphore等待的数据:
SELECT *
FROM sys.SYSPROCESSES
WHERE lastwaittype = 'RESOURCE_SEMAPHORE'
ORDER BY lastwaittype
由于这种情况不好模拟出来,所以没有截图。
步骤2:
从步骤1中得到的结果,可能会看到很多的事务处于ResourceSemaphore 等待状态,现在可以运行下面语句来查看已分配到内存的查询的目前状态,和未被分配内存的查询的数量。这个DMV会返回两行,一行是resource_semaphore_id为0的大查询,另外一些是为1的小查询,这里的小是内存小于5M。在这里可以获得总授予内存和实例上总可用内存。可以查看grantee_count和waiter_count,grantee_count是已经分配了内存的总查询数量,而waiter_count是在队列中等待授予内存的总查询数量:
SELECT *
FROM sys.dm_exec_query_resource_semaphores
步骤3:
然后使用DMV:sys.dm_exec_query_memory_grants来获得在等待队列中的查询所需要内存的详细信息。这些查询的grant_time和granted_memory_kb可能为null。也可以从这个DMV中得到plan_handle和sql_handle:
SELECT *
FROM sys.dm_exec_query_memory_grants
我们要关注的是下面3列:
步骤4:
现在将要找到集中消耗内存的查询,可以查看所有等待查询中的需求内存。当看到这部分内存太大的时候,然后找到这些查询的plan_handle,并查看它们的执行计划:
SELECT TOP 10
*
FROM sys.dm_exec_query_memory_grants
步骤5:
把步骤4中查询的plan_handle的数据复制,然后执行:
SELECT * FROM sys.dm_exec_sql_text(sql_handle)
注意替换括号中的sql_handle。然后查看其执行计划。
总结:
通过上面的步骤找到耗费内存的查询后,应该调整语句,使其占用更少的资源。以便解决内存压力。
分享到:
相关推荐
当一个SQLServer实例运行得很慢的时候,应该做一些检查,如检查等待状态。好的方法是一开始建立一个性能基线,以便做性能对比。当发现与性能基线对比后,存在内存压力的话,要找出是什么原因导致的。可以检查事务...
本文将介绍: a. 使用CMSIS API ,介绍FreeRTOS中计数信号量 b. 不使用CMSIS API,直接使用FreeRTOS函数 1. 简介 计数信号量可用于控制对资源的访问。要获得对资源的控制,任务必须首先获得信号量。...
线程间共享数据 读者写者问题 makefile
在多核DSP中 核间的同步问题使用硬件信号量来解决,本代码是一个硬件信号量的使用例程,对初学者有一些帮助
在windows系统下的os平台中,各个任务间是通过信号量来同步共同资源的,本文就信号量机制进行了详细的讲解,
ios开发GCD的dispatch_semaphore:1、常用函数及作用 2、信号量的用途:1》同步不同线程 2》当做锁来使用:3》控制同时执行的线程数:
释放信号量_tx_semaphore_put 1,如果tx_semaphore_suspension_list挂起队列为空,那么直接把tx_semaphore_count计数器加一 2,如果tx_semaphore_suspension_list挂起队列不为空,那么tx_semaphore_suspension_list...
Implement semaphore with mutex and conditional variable. Reason being, POSIX semaphore on Android are not used or well tested.
在Linux进程交互中,信号semaphore的使用。以及如何使用信号同步两个进程。
信号量基本上用于将任务与系统中的其他事件同步。在FreeRTOS中,信号量是基于队列机制实现的。FreeRTOS中有4种信号量: - 二进制信号量 - 计数信号量 - 互斥信号量 - 递归信号量 本文介绍: 二进制信号量的使用方法
学SOPC中uc/os II的基础代码,硬件配置好了,建立软件工程,把代码加进去就可以用了!
MINIX操作系统的内核修改补丁,增加了实时调度,迷你文件系统,semaphore系统服务。
This example demonstrates the use of Semaphores, Threads, and Timers. Three "Filler" threads are created that loop waiting for a while and then adding a value to the end of a circular buffer. ...
经典的信号量系列丛书, 豆瓣评分8.0, 在原版的基础上为PDF添加了目录, 更便于阅读
uCOS课件 ppt课件 介绍了操作系统基础 uCOS基础及任务同步 信号量 内存分配 uCOS移植
read_exclusive_semaphore
Football_using_semaphore 使用信号量和共享内存模拟足球比赛
多线程编程:信号量使用。 打包文件包含两个文件:c文件源代码、Makefile文件,运行环境在Ubuntu14.04下,使用自带的gcc编译器,同学们只需将文件夹复制到某一目录下之后在终端执行:1.“make”生成“test”可执行...
1、信号量:就是一种可用来控制访问资源的数量的标识,设定了一个信 2、信号量主要有3个函数,分别是: 3、那么就开头提的问题,我们用代码来解决
tx_semaphore_create, tx_semaphore_delete, tx_semaphore_get, tx_semaphore_info_get, tx_semaphore_prioritize, tx_semaphore_put. 线程相关API tx_thread_create, tx_thread_delete, tx_thread_identify, tx_...