SQL Server 环形缓冲区(Ring Buffer) -- RING_BUFFER_MEMORY_BROKER 诊断内部内存压力

内存Broker

内存Broker的职责是根据其需求在大内存消费者之间分配内存。内存Broker是一种SQLOS的组件,但是与缓冲池紧密结合。此外,内存Broker只会将缓冲池的内存管理器所控制的内存考虑在内。内存Broker会监视缓冲池的内存需求,以及由大内存消费者所消费的内存。基于所收集的信息,它会估计每个消费者的"最优化"内存分布,并将此信息广播给消费者。每个消费者会相应地使用此信息来适应其内存使用。

 

SQL Server 2005拥有4种主要的内存消费者:缓冲池的数据缓存、查询执行引擎(有时被称为工作台内存,workspace memory)、缓存以及编译引擎。在SQL Server 2005中,只有后3者会被内存Broker监视到。

 

用户可以通过查询内存Broker的环形缓冲,来监视内存Broker当前的、预计的以及推荐的内存分布。

 

select * from sys.dm_os_ring_bufferwherering_buffer_type = 'RING_BUFFER_MEMORY_BROKER'

 

注意,内存Broker的环形缓冲只在内存Broker希望某个特定组件的行为发生变化(如收缩、增长或者保持稳定)时才会被更新。

用户可以使用dbcc memorystatus来找到最后一个内存Broker的通知。

 

内部内存压力

当监测到内部内存压力时,为组件在缓存池分配内存的低内存通知将被打开。打开低内存通知允许从使用缓存池的高速缓存和其他组件中回收页面。

 

内部内存压力可以通过调整max server memory选项或当stolen页面与缓存池的比例超过80%时触发。

 

内部内存压力通知(‘Shrink’)能通过使用下列代码查询ring buffer的调用来发现。

 

selectx.value('(//Record/@time)[1]', 'bigint') as [Time Stamp],x.value('(//Notification)[1]', 'varchar(100)') as [Last Notification]from(select cast(record as xml)from sys.dm_os_ring_bufferswhere ring_buffer_type = 'RING_BUFFER_MEMORY_BROKER') as R(x)order by[Time Stamp] desc

 

 

以下为常用脚本:

 

/*RING_BUFFER_MEMORY_BROKER  Identity internal memory pressure.       Look at the 3 consumers of Memory     for SHRINK Notification          1.cache          2.query compilations          3.query executions      */select          CAST(record AS XML)          ,CAST(record AS XML).value('/Record[1]/@time[1]','bigint') as time          ,CAST(record as XML).value('/Record[1]/MemoryBroker[1]/Broker[1]','varchar(50)') as Broker          ,CAST(record as XML).value('/Record[1]/MemoryBroker[1]/Notification[1]','varchar(20)') as Notification          ,CAST(record as XML).value('/Record[1]/MemoryBroker[1]/MemoryRatio[1]','int') as MemoryRatio      from     sys.dm_os_ring_buffers     where           ring_buffer_type = 'RING_BUFFER_Memory_broker'       order by            CAST(record as XML).value('/Record[1]/MemoryBroker[1]/Broker[1]','varchar(50)') asc,            CAST(record as XML).value('/Record[1]/MemoryBroker[1]/Notification[1]','varchar(20)') asc