zoukankan      html  css  js  c++  java
  • SQL SERVER 作业浅析

    作业介绍

        SQL SERVER的作业是一系列由SQL SERVER代理按顺序执行的指定操作。作业可以执行一系列活动,包括运行Transact-SQL脚本、命令行应用程序、Microsoft ActiveX脚本、Integration Services 包、Analysis Services 命令和查询或复制任务。作业可以运行重复任务或那些可计划的任务,它们可以通过生成警报来自动通知用户作业状态,从而极大地简化了 SQL Server 管理[参见MSDN]。

        创建作业、删除作业、查看作业历史记录....等所有操作都可以通过SSMS管理工具GUI界面操作,有时候也确实挺方便的。但是当一个实例有多个作业或多个数据库实例时,通过图形化的界面去管理、维护作业也是个头痛的问题,对于SQL脚本与GUI界面管理维护作业熟优熟劣这个问题,只能说要看场合。下面主要介绍通过SQL脚本来管理、维护作业。

    作业分类

        创建作业时,往往需要指定作业类别,如果不指定新建作业类别,就会默认为“[未分类(本地)]”,如下截图所示:

    clipboard

        当然,你可以查看、添加、删除、修改作业分类。请看下面操作。

    1:查看作业分类

    Code Snippet
    1. --method 1:
    2. EXEC msdb.dbo.sp_help_category;
    3.  
    4. GO
    5.  
    6. --method 2:
    7.   SELECT  category_id            ,--作业类别ID
    8.           category_class         ,--类别中项目类型:1=作业2=警报 3=操作员
    9.           category_type          ,--类别中类型:=本地、=多服务器、=无
    10.           name                    --分类名称
    11.     FROMmsdb.dbo.syscategories

    有兴趣的可以研究一下存储过程msdb.dbo.sp_help_category

     sp_help_category

    2:添加作业分类

    如下所示,添加一个叫"DBA_MONITORING"的作业分类

    Code Snippet
    1. EXEC msdb.dbo.sp_add_category
    2.     @class=N'JOB',
    3.     @type=N'LOCAL',
    4.     @name=N'DBA_MONITORING' ;
    5.  
    6. GO
    7.  
    8.   
    9.  
    10. SELECT * FROM msdb.dbo.syscategories WHERE NAME='DBA_MONITORING'
    11.  
    12.  category_id category_class category_type    name
    13. ----------- -------------- ------------- -------------
    14. 102         1              1             DBA_MONITORING

    有兴趣的可以研究一下存储过程msdb.dbo.sp_add_category

     sp_add_category

    3:删除作业分类

    如下所示,删除一个叫"DBA_MONITORING" 的作业分类

    Code Snippet
    1. EXEC msdb.dbo.sp_delete_category
    2.     @name = N'DBA_MONITORING',
    3.     @class = N'JOB' ;
    4.  
    5. GO

    有兴趣的可以研究一下存储过程msdb.dbo.sp_delete_category

     sp_delete_category

     4:修改作业类别

    msdb.dbo.sp_update_category
         [@class =] 'class' , 
         [@name  =] 'old_name' ,
         [@new_name =] 'new_name'

    有兴趣的可以研究一下存储过程msdb.dbo.sp_update_category

     sp_update_category

    分析上面四个存储过程可以看出,实质上新增、修改、删除、查看作业类别无非就是对表 msdb.dbo.syscategories进行操作,只是通过存储过程封装了而已,增加了验证等操作,确保数据完整性。

    新建作业

    创建作业的步骤一般如下所示:

    1. 执行 sp_add_job 来创建作业。
    2. 执行 sp_add_jobstep 来创建一个或多个作业步骤。

    3. 执行 sp_add_schedule 来创建计划。

    4. 执行 sp_attach_schedule 将计划附加到作业。

    5. 执行 sp_add_jobserver 来设置作业的服务器。

    本地作业是由本地 SQL Server 代理进行缓存的。因此,任何修改都会隐式强制 SQL Server 代理重新缓存该作业。由于直到调用 sp_add_jobserver 时,SQL Server 代理才缓存作业,因此最后调用 sp_add_jobserver 将更为有效。

    下面看用脚本新建一个作业用来每天执行exec sp_cycle_errorlog ,实现错误日志循环, 从下面的脚本量来看,用脚本新建一个作业确实工作量很大,而且容易出错,GUI图形界面创建作业要方便得多,但是如果迁移数据库时,用脚本来新建作业是相当方便的。比GUI图形界面新建一个作业快捷方便多了。

    Code Snippet
    1. USE [msdb]
    2. GO
    3.  
    4. /****** Object:  Job [JOB_CYCLE_ERRORLOG]    Script Date: 08/23/2013 15:25:09 ******/
    5.   IFEXISTS(SELECT job_id FROM msdb.dbo.sysjobs_view WHERE name = N'JOB_CYCLE_ERRORLOG')
    6. EXEC msdb.dbo.sp_delete_job@job_id=N'a5dff08b-95f8-498e-a6c9-59241fe197b4', @delete_unused_schedule=1
    7. GO
    8.  
    9. USE [msdb]
    10. GO
    11.  
    12. /****** Object:  Job [JOB_CYCLE_ERRORLOG]    Script Date: 08/23/2013 15:25:09 ******/
    13. BEGIN TRANSACTION
    14. DECLARE @ReturnCode INT
    15. SELECT @ReturnCode = 0
    16. /****** Object:  JobCategory [DBA_MATIANCE]    Script Date: 08/23/2013 15:25:09 ******/
    17. IF NOT EXISTS(SELECT name FROM msdb.dbo.syscategories WHERE name=N'DBA_MATIANCE' AND category_class=1)
    18. BEGIN
    19. EXEC @ReturnCode = msdb.dbo.sp_add_category@class=N'JOB', @type=N'LOCAL', @name=N'DBA_MATIANCE'
    20. IF (@@ERROR <> 0 OR @ReturnCode <> 0) GOTO QuitWithRollback
    21.  
    22. END
    23.  
    24. DECLARE @jobId BINARY(16)
    25.   EXEC @ReturnCode =msdb.dbo.sp_add_job@job_name=N'JOB_CYCLE_ERRORLOG',
    26.         @enabled=1,
    27.         @notify_level_eventlog=0,
    28.         @notify_level_email=0,
    29.         @notify_level_netsend=0,
    30.         @notify_level_page=0,
    31.         @delete_level=0,
    32.         @description=N'每天执行exec sp_cycle_errorlog 实现错误日志循环。',
    33.         @category_name=N'DBA_MATIANCE',
    34.         @owner_login_name=N'sa', @job_id = @jobId OUTPUT
    35. IF (@@ERROR <> 0 OR @ReturnCode <> 0) GOTO QuitWithRollback
    36. /****** Object:  Step [Step 1: recycle the errorlog]    Script Date: 08/23/2013 15:25:09 ******/
    37. EXEC @ReturnCode = msdb.dbo.sp_add_jobstep@job_id=@jobId, @step_name=N'Step 1: recycle the errorlog',
    38.         @step_id=1,
    39.         @cmdexec_success_code=0,
    40.         @on_success_action=1,
    41.         @on_success_step_id=0,
    42.         @on_fail_action=2,
    43.         @on_fail_step_id=0,
    44.         @retry_attempts=0,
    45.         @retry_interval=0,
    46.         @os_run_priority=0, @subsystem=N'TSQL',
    47.         @command=N'exec msdb.dbo.sp_cycle_errorlog',
    48.         @database_name=N'msdb',
    49.         @flags=0
    50. IF (@@ERROR <> 0 OR @ReturnCode <> 0) GOTO QuitWithRollback
    51. EXEC @ReturnCode = msdb.dbo.sp_update_job @job_id = @jobId, @start_step_id = 1
    52. IF (@@ERROR <> 0 OR @ReturnCode <> 0) GOTO QuitWithRollback
    53. EXEC @ReturnCode = msdb.dbo.sp_add_jobschedule @job_id=@jobId, @name=N'Job Schedule',
    54.         @enabled=1,
    55.         @freq_type=4,
    56.         @freq_interval=1,
    57.         @freq_subday_type=1,
    58.         @freq_subday_interval=0,
    59.         @freq_relative_interval=0,
    60.         @freq_recurrence_factor=0,
    61.         @active_start_date=20130823,
    62.         @active_end_date=99991231,
    63.         @active_start_time=0,
    64.         @active_end_time=235959,
    65.         @schedule_uid=N'2099c694-cd26-4edf-8803-179227bf8770'
    66. IF (@@ERROR <> 0 OR @ReturnCode <> 0) GOTO QuitWithRollback
    67. EXEC @ReturnCode = msdb.dbo.sp_add_jobserver @job_id = @jobId, @server_name = N'(local)'
    68. IF (@@ERROR <> 0 OR @ReturnCode <> 0) GOTO QuitWithRollback
    69. COMMIT TRANSACTION
    70. GOTO EndSave
    71. QuitWithRollback:
    72.     IF (@@TRANCOUNT > 0) ROLLBACK TRANSACTION
    73. EndSave:
    74.  
    75. GO

    作业系统表

        与作业有关的系统表、视图大致有下面9个,下面就不费口舌详细解说每一个系统表的作用了,MSDN文档上有详细的解说,有兴趣的翻看一下即可。

    SELECT * FROM msdb.dbo.sysjobs            --存储将由 SQL Server 代理执行的各个预定作业的信息

    SELECT * FROM msdb.dbo.sysjobschedules    --包含将由 SQL Server 代理执行的作业的计划信息

    SELECT * FROM msdb.dbo.sysjobactivity;    --记录当前 SQL Server 代理作业活动和状态

    SELECT * FROM msdb.dbo.sysjobservers      --存储特定作业与一个或多个目标服务器的关联或关系

    SELECT * FROM msdb.dbo.sysjobsteps;       --包含 SQL Server 代理要执行的作业中的各个步骤的信息

    SELECT * FROM msdb.dbo.sysjobstepslogs;   --包含所有 SQL Server 代理作业步骤的作业步骤日志

    SELECT * FROM msdb.dbo.sysjobs_view;      --

    SELECT * FROM msdb.dbo.sysjobhistory      --包含有关 SQL Server 代理执行预定作业的信息

    SELECT * FROM msdb.dbo.syscategories      --包含由 SQL Server Management Studio 用来组织作业、警报和操作员的类别

    运行作业

    启动作业

            1:通过SSMS工具启动作业[参见MSDN]

            2:通过SQL命令启动作业

    启动作业一般通过sp_start_job来实现,具体语法与操作见下面。

    复制代码
    语法:
    
    sp_start_job 
         {   [@job_name =] 'job_name'
           | [@job_id =] job_id }
         [ , [@error_flag =] error_flag]
         [ , [@server_name =] 'server_name']
         [ , [@step_name =] 'step_name']
         [ , [@output_flag =] output_flag]
         
    例子:
    
    exec msdb.dbo.sp_start_job @job_name='JOB_CYCLE_ERRORLOG'
    复制代码

    停止作业

            1:通过SSMS工具停作业[参见MSDN]

            2:通过SQL命令停止作业

    复制代码
    语法:
    sp_stop_job 
          [@job_name =] 'job_name'
        | [@job_id =] job_id 
        | [@originating_server =] 'master_server'
        | [@server_name =] 'target_server'
    
    例子:
    exec msdb.dbo.sp_stop_job    @job_name='JOB_CYCLE_ERRORLOG'
    复制代码

     启用或禁用作业

            1:通过SSMS工具启用作业[参见MSDN]

            2:通过SQL命令禁用作业

    复制代码
    语法:
    sp_update_job [ @job_id =] job_id | [@job_name =] 'job_name'
         [, [@new_name =] 'new_name' ] 
         [, [@enabled =] enabled ]
         [, [@description =] 'description' ] 
         [, [@start_step_id =] step_id ]
         [, [@category_name =] 'category' ] 
         [, [@owner_login_name =] 'login' ]
         [, [@notify_level_eventlog =] eventlog_level ]
         [, [@notify_level_email =] email_level ]
         [, [@notify_level_netsend =] netsend_level ]
         [, [@notify_level_page =] page_level ]
         [, [@notify_email_operator_name =] 'email_name' ]
              [, [@notify_netsend_operator_name =] 'netsend_operator' ]
              [, [@notify_page_operator_name =] 'page_operator' ]
         [, [@delete_level =] delete_level ] 
         [, [@automatic_post =] automatic_post ]
    
    列子:
    EXEC msdb.dbo.sp_update_job @job_name = N'JOB_CYCLE_ERRORLOG', @enabled = 0 ; --0 禁用作业、 1启用作业 GO
    复制代码

    删除作业

    1:通过SSMS工具删除作业[参见MSDN]

    2:通过SQL命令删除作业

    复制代码
    语法:
    
    sp_delete_job { [ @job_id = ] job_id | [ @job_name = ] 'job_name' } , [ , [ @originating_server = ] 'server' ] [ , [ @delete_history = ] delete_history ] [ , [ @delete_unused_schedule = ] delete_unused_schedule ]
    
    例子:
    EXEC msdb.dbo.sp_delete_job  @job_name = 'JOB_CYCLE_ERRORLOG';
        
    复制代码

    迁移作业

    使用 Transact-SQL 编写作业脚本
    1. 对象资源管理器中,连接到 Microsoft SQL Server 数据库引擎实例,再展开该实例。

    2. 展开“SQL Server 代理”,再展开“作业”,然后右键单击要编写脚本的作业。

    3. 从快捷菜单中,选择“编写作业脚本为”,再选择“CREATE 到”“DROP 到”,并单击下列内容之一:

      新查询编辑器窗口,将打开一个新的查询编辑器窗口,并为其编写 Transact-SQL 脚本。

      文件,将 Transact-SQL 脚本保存到文件。

          剪贴板,将 Transact-SQL 脚本保存到剪贴板

    常用管理作业SQL

    1:查看属于某个数据库的所有作业。

    Code Snippet
    1. SELECT j.job_id        AS JOB_ID            ,
    2.        name            AS JOB_NAME          ,
    3.        enabled         AS JOB_ENABLED       ,
    4.        description     AS JOB_DESCRIPTION   ,
    5.        date_created    AS DATE_CREATED      ,
    6.        date_modified   AS DATE_MODIFIED
    7. FROM msdb.dbo.sysjobs j
    8. WHERE   job_id IN( SELECTjob_id
    9.                   FROM    msdb.dbo.sysjobsteps
    10.                   WHERE   database_name = 'DataBaseName' )

    2:查看某个作业类别的所有作业

    Code Snippet
    1. SELECT  j.name             AS Job_Name       ,
    2.          j.description      AS Job_Description ,
    3.          j.date_created     AS Date_Created    ,
    4.          j.date_modified    AS Date_Modified   ,
    5.          c.name             AS Job_Class
    6. FROM  msdb.dbo.sysjobs j
    7.       LEFT JOIN msdb.dbo.syscategories c ON j.category_id = c.category_id
    8.  WHEREc.name = '[Uncategorized (Local)]'

    3:查看禁用/启用的作业

    SELECT * FROM msdb.dbo.sysjobs WHERE  enabled=0    --0:禁用 1:为启用

    4:查看出错的作业记录

    4.1:查询那些作业在今天出错(如果要查询历史出错作业,去掉查询时间条件即可)

    Code Snippet
    1. SELECT name               AS JOB_NAME        ,
    2.       description         AS JOB_Description ,
    3.       date_created        AS Date_Created    ,
    4.       date_modified       AS Date_Modified
    5. FROM  msdb.dbo.sysjobs
    6. WHERE enabled = 1
    7.       AND job_id IN(
    8.       SELECT  job_id
    9.       FROM    Msdb.dbo.sysjobhistory
    10.       WHERE   run_status = 0
    11.               AND run_date = CAST(CONVERT(VARCHAR(8), GETDATE(), 112) AS INT) )

    4.2:查看出错详细信息

    Code Snippet
    1. SELECT j.name                AS JOB_NAME ,
    2.       h.step_id              AS STEP_ID  ,
    3.       h.step_name            AS STEP_NAME,
    4.       h.message              AS ERR_MSG  ,
    5.       h.run_date             AS RUN_DATE ,
    6.       h.run_time             AS RUN_TIME ,
    7.       msdb.dbo.agent_datetime(h.run_date, h.run_time) AS 'RunDateTime' ,
    8.       CAST(run_duration / 10000 AS VARCHAR(2)) + N'小时'
    9.     + CAST(( run_duration - run_duration / 10000 * 10000 ) / 100 AS VARCHAR(2))
    10.     + N'分钟' + SUBSTRING(CAST(run_duration AS VARCHAR(10)),
    11.                           LEN(CAST(run_duration AS VARCHAR(10))) - 1, 2)
    12.     + N'秒' AS run_duration
    13. FROM  msdb.dbo.sysjobhistory h
    14.       LEFT JOIN msdb.dbo.sysjobs j ON h.job_id = j.job_id
    15. WHERE run_status = 0
    16.       AND run_date = CAST(CONVERT(VARCHAR(8), GETDATE(), 112) AS INT)

    5:查看作业的执行时间:

    5.1:查看当天成功执行的作业的时间(查看的是作业Step信息)

    Code Snippet
    1. SELECT j.name         AS job_name ,
    2.       h.step_id      AS step_id  ,
    3.       h.step_name     AS step_name,
    4.       h.message       AS Message  ,
    5.       h.run_date      AS Run_date ,
    6.       h.run_time      AS run_time ,
    7.       msdb.dbo.agent_datetime(h.run_date, h.run_time) AS 'RunDateTime' ,
    8.       CAST(run_duration / 10000 AS VARCHAR(2)) + N'小时'
    9.       + CAST(( run_duration - run_duration / 10000 * 10000 ) / 100 AS VARCHAR(2))
    10.       + N'分钟' + SUBSTRING(CAST(run_duration AS VARCHAR(10)),
    11.                           LEN(CAST(run_duration AS VARCHAR(10))) - 1, 2)
    12.       + N'秒' AS run_duration
    13. FROM msdb.dbo.sysjobhistory h
    14.       LEFT JOIN msdb.dbo.sysjobs j ON h.job_id = j.job_id
    15.  WHERE run_status = 1
    16.       AND run_date = CAST(CONVERT(VARCHAR(8), GETDATE(), 112) AS INT)
    17.       ORDER BY run_duration DESC

    5.2:查询每个作业的执行时间、按执行时间降序

    Code Snippet
    1. SELECT  j.name            AS JOB_NAME ,
    2.         h.run_date        AS RUN_DATE ,
    3.         SUM(run_duration) AS SUM_DURATION
    4. FROM  msdb.dbo.sysjobhistory h
    5.         LEFT JOIN msdb.dbo.sysjobs j ON h.job_id = j.job_id
    6. WHERE run_status = 1
    7.         AND run_date = CAST(CONVERT(VARCHAR(8), GETDATE(), 112) AS INT)
    8. GROUP BY name ,
    9.         run_date
    10. ORDER BY Sum_Duration DESC

    参考资料:

    http://blog.csdn.net/dba_huangzj/article/details/8300178

  • 相关阅读:
    GCD多线程使用
    高德地图引入库错误std::string::find_first_of(char const*, unsigned long, unsigned long) const"
    vim配置 高亮+自动缩进+行号+折叠+优化
    设置MAC 下 Vim 语法高亮显示
    The platform of the target `Pods` (iOS 4.3) is not compatible 错误
    使用GCD创建单例
    使用Draw rect 绘制圆角矩形
    使用第三方类、库需要注意的正则类RegexKitLite的使用
    NSDate 时区转换问题
    UISlider设置按钮透明
  • 原文地址:https://www.cnblogs.com/zhangyingai/p/7082528.html
Copyright © 2011-2022 走看看