SQL Server(TSQL) - 是否可以并行执行EXEC语句?

时间:2010-12-31 18:57:39

标签: sql sql-server tsql

SQL Server 2008 R2

这是一个简化的例子:

EXECUTE sp_executesql N'PRINT ''1st '' + convert(varchar, getdate(), 126) WAITFOR DELAY ''000:00:10'''
EXECUTE sp_executesql N'PRINT ''2nd '' + convert(varchar, getdate(), 126)'

第一个语句将在继续之前打印日期并延迟10秒。 第二个陈述应立即打印。

T-SQL的工作方式,第二个语句在第一个语句完成之前不会被评估。如果我将其复制并粘贴到新的查询窗口,它将立即执行。

问题是我还有其他更复杂的事情,需要将变量传递给两个程序。

我想做的是:

  • 获取记录
  • 将其锁定一段时间
  • 当它被锁定时,对该记录和表本身执行一些其他语句

也许有办法动态创建一些工作?

无论如何,我正在寻找一种简单的方法,无需手动PRINT语句并复制/粘贴到另一个会话。

有没有办法在没有等待/并行的情况下执行EXEC?

6 个答案:

答案 0 :(得分:17)

是的,有办法,请参阅Asynchronous procedure execution

但是,这可能不是你需要的。 T-SQL是一种数据访问语言,当您考虑事务时,锁定和提交/回滚语义几乎不可能具有并行作业。并行T-SQL适用于requests queues,其中每个请求都是独立的,并且作业之间没有相关性。

你所描述的内容完全听起来不像是能够,也不应该实际上被贬低的东西。

答案 1 :(得分:2)

如果要锁定记录以便可以对其执行语句,则可能需要将这些语句作为事务执行。

要并行执行SQL,您需要通过Java,C ++,perl或任何其他编程语言中的单独线程/进程执行SQL来解析SQL调用(地狱,在后台的shell脚本中启动“isql”)会工作)

答案 2 :(得分:1)

查看文章Asynchronous T-SQL Execution Without Service Broker可能是值得的。

答案 3 :(得分:0)

如果在阅读完所有关于潜在问题的所有内容并且您仍希望并行运行之后,您可能可以尝试sql作业,将查询放在不同的作业中,然后通过调用此类作业来执行

EXEC msdb..sp_start_job 'Job1'

EXEC msdb..sp_start_job 'Job2' 

答案 4 :(得分:0)

SQL代理作业可以并行运行,也可以直接从TSQL创建。 answer by Remus Rusanu包含一个提及此问题的链接以及一些缺点。

另一个缺点是创建作业需要额外的安全权限。此外,对于下面的实现,作业必须作为具有其他作业管理权限的特定用户+登录运行。

可以将任意SQL作为一个不同的(更安全的)用户运行,但我认为它需要sysadmin权限来指定作业。

如果需要,返回的@pJobIdHexOut可用于停止作业。

create proc [Common].[usp_CreateExecuteOneTimeBackgroundJob] 
    @pJobNameKey          varchar(100),     -- Caller should ensure uniqueness to avoid a violation
    @pJobDescription      varchar(1000),
    @pSql                 nvarchar(max),
    @pJobIdHexOut         varchar(100) = null out, -- JobId as Hex string. For SqlServer 2014 binary(16) = varchar(64)
    @pDebug               bit = 0  -- True to include print messages
--
with execute as 'TSqlBackgroundJobOwner' -- requires special permissions (See below)
as
/*---------------------------------------------------------------------------------------------------------------------
    Purpose:  Create a one time background job and launch it immediately.  The job is owned by the "execute as" UserName 

              Caller must ensure the @pSql argument is safe.

Required Permissions for "execute as" user:

        -- User must be created with associated login (w/ deny connect).

        use [msdb];
        create user [$UserName$] for login [$LoginName$];
        alter role [SQLAgentUserRole] add member [$UserName$];
        alter role [SQLAgentReaderRole] add member [$UserName$];
        alter role [SQLAgentOperatorRole] add member [$UserName$];
        grant select on dbo.sysjobs to [$UserName$];
        grant select on dbo.sysjobactivity to [$UserName$];',

        use [Master];
        create user [$UserName$] for login [$LoginName$];
        grant execute on xp_sqlagent_is_starting to [$UserName$];
        grant execute on xp_sqlagent_notify to [$UserName$];';


    Modified    By           Description
    ----------  -----------  ------------------------------------------------------------------------------------------
    2014.08.22  crokusek     Initial version   
    2015.12.22  crokusek     Use the SP caller as the job owner (removed the explicit setting of the job owner).
  ---------------------------------------------------------------------------------------------------------------------*/
begin try       

    declare
        @usp                  varchar(100) = object_name(@@procid),
        @currentDatabase      nvarchar(100) = db_name(),
        @jobId                binary(16),        
        @jobOwnerLogin        nvarchar(100);

    set xact_abort on;    -- ensure transaction is aborted on non-catchables like client timeout, etc.
    begin transaction

        exec msdb.dbo.sp_add_job 
            @job_name=@pJobNameKey,
                @enabled=1, 
                @notify_level_eventlog=0, 
                @notify_level_email=2, 
                @notify_level_netsend=2, 
                @notify_level_page=2, 
                @delete_level=3, 
                @description=@pJobDescription,
                @category_name=N'Database Maintenance',
            -- If not overridden then the the current login is the job owner
            --@owner_login_name=@jobOwnerLogin,  -- Requires sysadmin to set this so avoiding.
            @job_id = @jobId output;

        -- Get the job_id string of the jobId (special format)
        --
        set @pJobIdHexOut = Common.ufn_JobIdFromHex(@jobId);

        if (@pDebug = 1)
        begin
            print 'JobId: ' + @pJobIdHexOut;
            print 'Sql: ' + @pSql;
        end

        exec msdb.dbo.sp_add_jobserver @job_id=@jobId; -- default is local server

        exec msdb.dbo.sp_add_jobstep 
            @job_id=@jobId, 
            @step_name=N'One-Time Job Step 1', 
                @step_id=1, 
            @command=@pSql,
                @database_name=@currentDatabase,
                @cmdexec_success_code=0, 
                @on_success_action=1, 
                @on_fail_action=2, 
                @retry_attempts=0, 
                @retry_interval=0, 
                @os_run_priority=0,
            @subsystem=N'TSQL', 
                @flags=0
            ;

          declare
              @startResult int;                    

          exec @startResult = msdb.dbo.sp_start_job 
              @job_id = @jobId;

      -- End the transaction
      --
      if (@startResult != 0)          
          raiserror('Unable to start the job', 16, 1);  -- causes rollback in catch block
      else
          commit;   -- Success

end try
begin catch
    declare        
        @CatchingUsp  varchar(100) = object_name(@@procid);    

    if (xact_state() = -1)
        rollback;

    --exec Common.usp_Log
    --    @pMethod = @CatchingUsp;

    --exec Common.usp_RethrowError
    --    @pCatchingMethod = @CatchingUsp;
end catch

答案 5 :(得分:0)

您可以创建一个具有2个并行运行的任务的SSIS。然后进行计划外的代理作业以调用此SSIS。您最终可以使用sp_start_job执行此计划外的代理作业。