微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

sql server中的任务调度与CPU深入讲解

一. 概述

我们知道在操作系统看来, sql server产品与其它应用程序一样,没有特别对待。但内存,硬盘,cpu又是数据库系统最重要的核心资源,所以在sql server 2005及以后出现了sqlOS,这个组件是sqlserver和windows的中间层,用于cpu的任务调度,解决I/O的资源争用,协调内存管理等其它的资源协调工作。下面我来试着讲讲sqlOS下的Scheduler调度管理。话不多说了,来一起看看详细的介绍吧。

二. cpu 的配置

sql server 里点击数据库实例右键到属性,选择处理器进行配置。最大工作线程数的认值是0 (注意这里配置的是worker它是对cpu的真正封装)。这使得sql Server能够在启动时自动配置工作线程的数量认设置对于大多数系统是最好的。但是,根据您的系统配置,将最大工作线程数设置为一个特定的值有时会提高性能。当查询请求的实际数量小于最大工作线程数时,一个线程处理一个查询请求。但是,如果查询请求的实际数量超过最大线程量时,sqlServer会将Worker Threads线程池化,以便下一个可用的工作线程可以处理请求。

配置如下图所示:

也可以通过T-sql配置,下例通过sp_configure将max worker线程选项配置为900

rush:sql;"> USE AdventureWorks2012 ; GO EXEC sp_configure 'show advanced options',1; GO RECONfigURE ; GO EXEC sp_configure 'max worker threads',900 ; GO RECONfigURE;

Max Worker Threads服务器配置选项不考虑的线程, 像高可用、Service broker、 Lock 管理等其它。如果配置的线程数量超过了,下面的查询将提供关于系统任务产生的额外线程信息

is_user_process = 0 表示系统任务,非用户任务。

rush:sql;"> SELECT s.session_id,r.command,r.status,r.wait_type,r.scheduler_id,w.worker_address,w.is_preemptive,w.state,t.task_state,t.session_id,t.exec_context_id,t.request_id FROM sys.dm_exec_sessions AS s INNER JOIN sys.dm_exec_requests AS r ON s.session_id = r.session_id INNER JOIN sys.dm_os_tasks AS t ON r.task_address = t.task_address INNER JOIN sys.dm_os_workers AS w ON t.worker_address = w.worker_address WHERE s.is_user_process = 0;

下面显示每个用户的活动会话数

rush:sql;"> SELECT login_name,COUNT(session_id) AS session_count FROM sys.dm_exec_sessions WHERE status<>'sleeping' GROUP BY login_name;

下表显示了各种cpusqlServer组合的最大工作线程的自动配置数量

Number of cpus

32-bit computer

64-bit computer

<= 4 processors

256

512

8 processors

288

576

16 processors

352

704

32 processors

480

960

64 processors

736

1472

128 processors

4224

4480

256 processors

8320

8576

根据微软的建议:这个选项是一个高级选项,应该只由经验丰富的数据库管理员或经过认证的sql Server专业人员更改。如果您怀疑存在性能问题,则可能不是工作线程的可用性。原因更像是I/O,这会导致工作线程等待。在更改最大工作线程设置之前,最好找到性能问题的根本原因。

二.调度原理

2.1 Scheduler任务调度

sqlserver 的一个Scheduler对应操作系统上的一个逻辑cpu用于任务分配。调度分配从NUMA节点级别开始。基本算法是一个用于新连接的循环调度。当每个新的连接到达时,它被分配给基于循环的调度器。在相同的NUMA节点内,以最小的负载因子分配给调度器的新连接。

2.2 Worker

Worker又称为WorkerThread,每个Worker跟一个线程,是sql server任务的执行单位。 多个Worker对应一个Scheduler,公式Workers=max worker threads/onlines scheduler。在一个Scheduler上,同一时间只能有一个Worker运行。例如4个处理器的64位操作系统,它的每个Scheduler的Worker是512/4=128。

2.3 Task

在Worker上运行的最小任务单元。最简单的Task就是一个简单的Batch,当一个会话发出一个请求时,sql server会把这个请求拆分一个或多个任务(Tasks),然后关联对应个数的工作者线程(worker thread)。

例如下面是二个Task,二个Task可能不是同一个Worker。二个Worker也可能不是同一个Scheduler.

rush:sql;"> select @@servername Go select getdate() GO

每个Task线程都有3个状态:

  • Running: 一个处理器在某个时间只能做一件事情,当一个线程正在一个处理器上运行时,这个线程的状态就是running。
  • Suspended: 没有足够资源时,当前线程放弃占有处理器,变成挂起状态。
  • Runnable: 一个线程已完成了等待,但还没有轮到它运行,就会变成runnable状态,这种信号等待(signal wait)

2.4 Yielding

Yelding就是所有逻辑scheduler上运行的Worker都是非抢占式的,在 Scheduler上Worker由于资源等待,让出给其它Worker就叫Yielding。

下面讲述几种发生的状态:

  1. 当Woker在Scheduler上运行了超过4ms,就做Yielding。

  2. 每做64k的结果集的排序,就会做一次Yielding。

  3. 做语句Complie编译的过程中,这个过程比较占cpu资源时,经常会有Yielding等。

2.5 调度关系图如下:

2.5 Task在调度运行图如下:

  

1. 当 Task 是Runnig时,它是Schedler的活动Worker。

2. 当 Task只等待cpu运行时,它被放入Schedler可运行的队列中。

3. 当 Task 在等待某个资源时(比如锁、磁盘输入/输出等)时,它处于“Suspended挂起状态” 状态。

4. 如果Task Scheduler挂起状态完成了等待,那么它就会被放到Scheduler 的Runnable队列的末尾。

5. 如果运行线程自动Yidlding让步,则将其放回Scheduler 的Runnable队列的末尾。

6. 如果运行的线程需要等待某个资源,它将被调出Scheduler调度器并进入挂起状态Waiter list。

7. 如果正在运行的线程完成它的工作,那么Runnable队列的顶部的第一个线程就变成了“运行”线程。

三. 使用dmv任务查看

3.1. 通过sys.dm_os_sys_info 查看scheduler与cpu的关系如下:

rush:sql;">  SELECT cpu_count,max_workers_count,scheduler_count FROM sys.dm_os_sys_info

  

3.2 查看最大Worker数  

rush:sql;"> select max_workers_count from sys.dm_os_sys_info  

3.3 查看Task与Worker关系

50

select state,last_wait_type,tasks_processed_count,task_address,worker_address,scheduler_address
from sys.dm_os_workers where worker_address =0x00000000043621A0

3.4 查看Scheduler

rush:sql;"> --scheduler_id<255 代表用户cpu,相反代表SYstem SCHEDULER SELECT scheduler_id,cpu_id,is_online,current_tasks_count,runnable_tasks_count,current_workers_count,active_workers_count,work_queue_count FROM sys.dm_os_schedulers WHERE scheduler_id < 255

cpu_id:关联的cpucpu ID >=255 这类Scheduler都用于系统内部使用。比如说资源管理、DAC、备份还原操作等。

  is_online: 0 调度器离线,1 在线。

  current_tasks_count:当前任务数,状态包括:(等待,运行,已完成)。

  runnable_tasks_count:以分配任务,并在可运行队列中等待被调度的任务数,使用率不高的情况下,这个值会是0。

  current_workers_count:此scheduler关联的线程数。包括处于空闲状态的线程work。

  active_workers_count:当前处理活动的线程数,它必须关联任务task,包括running,runnable,suspend。

  work_queue_count:队列中的任务task等待数,如果不为0,意味着线程用尽的压力。

讲到这里,后面讲讲cpuf过高的分析...

参考文献:

  

  Microsoft SQL Server企业级平台管理实践

  

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对编程之家的支持

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


SELECT a.*,b.dp_name,c.pa_name,fm_name=(CASE WHEN a.fm_no=&#39;LJCG001H&#39; THEN dbo.ELTPNAME(a.fw_nu) ELSE d.fm_name END),e.fw_state_nm,f.fw_rmk_nm
if not exists(select name from syscolumns where name=&#39;tod_no&#39; and id=object_id(&#39;iebo09d12&#39;)) alter table iebo09d12 add tod_no varchar(
select a.*,pano=a.pa_no,b.pa_name,f.dp_name,e.fw_state_nm,g.fa_name from LJSS007H a (nolock) Left join LJPA002H b (nolock) On a.pa_no =b.pa_no Left jo
要在 SQL Server 2019 中设置定时自动重启,可以使用 Windows 任务计划程序。下面是详细的步骤: 步骤一:创建批处理文件 打开记事本。 输入以下内容: net stop &quot;SQL Server (MSSQLSERVER)&quot; net start &quot;SQ
您收到的错误消息表明数据库 &#39;EastRiver&#39; 的事务日志已满,导致数据库操作失败。要解决这个问题,可以按照以下步骤操作: 1. 备份事务日志首先,备份事务日志以释放空间: BACKUP LOG [EastRiver] TO DISK = N&#39;C:\Backup\East
首先我需要查询出需要使用SQL Server Profiler跟踪的数据库标识ID,若不知道怎么查询数据库的标识ID, 打开SQL Server management studio,点击工具。选择SQL Server Profiler。 登录,登录成功后,如果有个默认弹窗,先取消 新建追踪 命名
--最新的解决方法 --先创建用户帐户,不进行授权,然后通过下面的SQL语句将该用户帐户关联至对应的数据库用户。优点是避免了重新授权的操作。 USE tempdbEXEC sp_change_users_login &#39;Update_One&#39;, &#39;iemis&#39;, &#3
命令: ALTER TABLE 表名 add 列名 数据类型 default 默认值 not null 例如: ALTER TABLE LJEL005H add el_req int default 15 not null
declare @i int set @i=340 while @i&lt;415 begin set @i=@iʱ insert into LJWK007H select &#39;2024&#39;,&#39;28&#39;,&#39;9110&#39;,&#39;3PTSD621000000