2012-08-16 108 views
0

可能重複:
Stored procedure without cursors編寫存儲過程,而不光標

我怎麼能不寫光標下面的SP?更重要的是它沒有給我期望的輸出。我沒有寫這個,我試圖解釋這是什麼問題。

ALTER PROCEDURE [dbo].[AccreditationExpiryCheck] 
AS 
BEGIN 
    SET NOCOUNT ON; 

    declare @taskTypeId int = 19 -- Accreditations, automated 
    declare @firstActionTypeId int = 23 -- Accreditation expiring 
    declare @nextActionTypeId int = 3 -- Call company 

    declare @companyId int 
    declare @accreditationId int 
    declare @comment nvarchar(max) = N' accreditation for this company has expired.' 

    -- find all companies and accreditations expiring 
    declare companies cursor local forward_only read_only for 
     select c.Company_Id, a.Accred_ID 
     from COMPANY c 
      inner join MEMBERSHIP m on c.Company_ID = m.Company_ID 
      inner join ACCREDITATION a on c.Company_ID = a.Company_ID 
     where 
      -- Accreditation expired yesterday 
      cast(a.Accred_ExpDate as DATE) = cast(DATEADD(DAY, -1, GETDATE()) as DATE) 
      and m.IsMember_Ind = 1 
      and (c.HQ_ID IS NULL OR c.HQ_ID = c.Company_ID) -- FB4640: this isn't a 'team' co (with an HQ) 
      -- and there is no action of this type created within 1 day 
      -- of the expiry date 
      and not exists (
       select * from TaskAction ta where 
        ta.FirstActionTypeId = @firstActionTypeId and 
        ta.TaskTypeId = @taskTypeId and 
        ta.TaskCreatedOn BETWEEN a.Accred_ExpDate AND DATEADD(DAY, 1, a.Accred_ExpDate) and 
        ta.EntityId = c.Company_ID and 
        ta.EntityTypeId = 1) 

    open companies 

    fetch next from companies into @companyId, @accreditationId 

    declare @title nvarchar(max) = 
     (select AccredType_Name from ACCREDITATION_TYPE at 
     inner join ACCREDITATION a on at.AccredType_ID = a.AccredType_ID 
     where a.Accred_ID = @accreditationId) 

    declare @comment2 nvarchar(max) = isnull(@title, '') + ' accreditation for this company has expired.' 
    while @@FETCH_STATUS = 0 
    begin 
     exec CreateSystemTask 
      @taskTypeId, 
      @firstActionTypeId, 
      @nextActionTypeId, 
      @companyid, 
      @comment2, 
      @title 

     fetch next from companies into @companyId,@accreditationId 
    end 

    close companies 
    deallocate companies 
END 

以上來自上述sp的select語句給了我正確的數據集,但循環的遊標給了我不同的輸出。

select c.Company_Id, a.Accred_ID 
     from COMPANY c 
      inner join MEMBERSHIP m on c.Company_ID = m.Company_ID 
      inner join ACCREDITATION a on c.Company_ID = a.Company_ID 
     where 
      -- Accreditation expired yesterday 
      cast(a.Accred_ExpDate as DATE) = cast(DATEADD(DAY, -1, GETDATE()) as DATE) 
      and m.IsMember_Ind = 1 
      and (c.HQ_ID IS NULL OR c.HQ_ID = c.Company_ID) -- FB4640: this isn't a 'team' co (with an HQ) 
      -- and there is no action of this type created within 1 day 
      -- of the expiry date 
      and not exists (
       select * from TaskAction ta where 
        ta.FirstActionTypeId = @firstActionTypeId and 
        ta.TaskTypeId = @taskTypeId and 
        ta.TaskCreatedOn BETWEEN a.Accred_ExpDate AND DATEADD(DAY, 1, a.Accred_ExpDate) and 
        ta.EntityId = c.Company_ID and 
        ta.EntityTypeId = 1) 
+2

除非我們可以內聯「CreateSystemTask」的定義,否則我們將無法改進它,除非自動爲行集中的每一行執行存儲過程。 – 2012-08-16 08:45:27

+1

請儘量避免重新發布您的問題。如果您需要對它們進行更改,則有一個編輯按鈕。 – 2012-08-16 08:51:08

+0

對不起。 – Joshua 2012-08-16 08:51:50

回答

0

您可以使用while循環去掉遊標,如下圖所示。如果你正在查詢的數據包含一個唯一的標識符,你可以不使用臨時表,但是對於你的例子,我使用了一個臨時表並且包含了一個recordId。

ALTER PROCEDURE [dbo].[AccreditationExpiryCheck] 
AS 
BEGIN 
    SET NOCOUNT ON; 

    declare @taskTypeId int = 19 -- Accreditations, automated 
    declare @firstActionTypeId int = 23 -- Accreditation expiring 
    declare @nextActionTypeId int = 3 -- Call company 

    declare @companyId int 
    declare @accreditationId int 
    declare @comment nvarchar(max) = N' accreditation for this company has expired.' 

    -- find all companies and accreditations expiring 
    select ROW_NUMBER() OVER(ORDER BY c.Company_Id, a.Accred_ID) as [RecordId], c.Company_Id as [Company_Id], a.Accred_ID as [Accred_ID] 
    into #COMPANIES 
     from COMPANY c 
      inner join MEMBERSHIP m on c.Company_ID = m.Company_ID 
      inner join ACCREDITATION a on c.Company_ID = a.Company_ID 
     where 
      -- Accreditation expired yesterday 
      cast(a.Accred_ExpDate as DATE) = cast(DATEADD(DAY, -1, GETDATE()) as DATE) 
      and m.IsMember_Ind = 1 
      and (c.HQ_ID IS NULL OR c.HQ_ID = c.Company_ID) -- FB4640: this isn't a 'team' co (with an HQ) 
      -- and there is no action of this type created within 1 day 
      -- of the expiry date 
      and not exists ( 
       select * from TaskAction ta where 
        ta.FirstActionTypeId = @firstActionTypeId and 
        ta.TaskTypeId = @taskTypeId and 
        ta.TaskCreatedOn BETWEEN a.Accred_ExpDate AND DATEADD(DAY, 1, a.Accred_ExpDate) and 
        ta.EntityId = c.Company_ID and 
        ta.EntityTypeId = 1) 

    declare @recordId int = 0; 
    declare @title nvarchar(max); 
    declare @comment2 nvarchar(max); 

    while(1=1) 
     begin 
      select top 1 @recordId = [RecordId] 
         ,@companyId = [CompanyId] 
         ,@accreditationId = [Accred_ID] 
      from #COMPANIES 
      where [RecordId] > @recordId 

      if @@ROWCOUNT = 0 break; 

      set @title = 
       (select AccredType_Name from ACCREDITATION_TYPE at 
       inner join ACCREDITATION a on at.AccredType_ID = a.AccredType_ID 
       where a.Accred_ID = @accreditationId) 

      set @comment2 = isnull(@title, '') + ' accreditation for this company has expired.' 

       exec CreateSystemTask 
        @taskTypeId, 
        @firstActionTypeId, 
        @nextActionTypeId, 
        @companyid, 
        @comment2, 
        @title   
     end 

    drop table #COMPANIES 
END