既然sqlserver内存有那么多种,每种都可能有各自上限值,DBA也必须能够看到sqlserver每种内存到底使用了多少,究竟是哪一种接近了上限、是哪部分内存不足,才能更好地解决问题。
通常可以用两种方法看到各部分内存用量——内存相关计数器和DMV视图
一、 内存相关计数器 与sqlserver相关的计数器通常以SQLServer:或MSSQL&
1. SQLServer:Memory Manager 总体内存使用情况
文章图片
下面是sqlserver各部分内存使用情况:
2. SQLServer:Buffer Manager 数据页读写情况
buffer pool是sqlserver内存使用最多也最容易出现瓶颈的部分,因此这部分计数器非常重要。
文章图片
【sqlserver|SqlServer 内存篇(三)—— SqlServer内存使用状况分析】
二、 动态性能视图DMV sqlserver使用Memory Clerk方式统一管理内存分配和回收,而跟踪内存使用最常用的视图也就叫做。
1. sqlserver各部分内存使用情况 —— sys.dm_os_memory_clerks
selecttype
, sum(virtual_memory_reserved_kb) VM_Reserved
, sum(virtual_memory_committed_kb) VM_Commited
, sum(awe_allocated_kb) AWE_Allocated
, sum(shared_memory_reserved_kb) Shared_Reserved
, sum(shared_memory_committed_kb) Shared_Commited
--, sum(single_pages_kb)--SQL2005、2008
--, sum(multi_pages_kb)--SQL2005、2008
fromsys.dm_os_memory_clerks
group by type
order by type;
字段含义如下:
文章图片
主要type如下:
文章图片
文章图片
文章图片
文章图片
文章图片
2. sqlserver缓存了哪些对象 —— sys.dm_os_buffer_descriptors
-- 查询当前数据库缓存的所有数据页面,哪些数据表,缓存的数据页面数量
-- 从这些信息可以看出,系统经常要访问的都是哪些表,有多大?
select p.object_id, object_name=object_name(p.object_id), p.index_id, buffer_pages=count(*)
from sys.allocation_units a,
sys.dm_os_buffer_descriptors b,
sys.partitions p
where a.allocation_unit_id=b.allocation_unit_id
and a.container_id=p.hobt_id
and b.database_id=db_id()
group by p.object_id,p.index_id
order by buffer_pages desc;
3. 执行计划都缓存了什么 —— sys.dm_exec_cached_plans
文章图片
-- 查询缓存的各类执行计划,及分别占了多少内存
-- 可以对比动态查询与参数化SQL(预定义语句)的缓存量
selectcacheobjtype
, objtype
, sum(cast(size_in_bytes as bigint))/1024 as size_in_kb
, count(bucketid) as cache_count
fromsys.dm_exec_cached_plans
group by cacheobjtype, objtype
order by cacheobjtype, objtype
查询结果会很大,注意将结果集输出到表或文件中,直接输出到DB服务器的SMSS可能导致资源争用
-- 查询缓存中具体的执行计划,及对应的SQL
-- 将此结果按照数据表或SQL进行统计,可以作为基线,调整索引时考虑
SELECTusecounts ,
refcounts ,
size_in_bytes ,
cacheobjtype ,
objtype ,
TEXT
FROMsys.dm_exec_cached_plans cp
CROSS APPLY sys.dm_exec_sql_text(plan_handle)
ORDER BY objtype DESC ;