在游标SELECT语句中使用变量

问题描述 投票:0回答:3

我编写了一个存储过程,它将采用父部件,遍历该部件的物料清单记录,并有条件地连接匹配的子记录(即包含在该部件的 BOM 中的部件),但只有子记录-满足特定条件的记录(使用我们的命名约定,以“.”、“E”或“ZG”开头的任何内容)。

我已经创建了 SP,它工作得很好,但只有当我传递[可选]父部件号时。我需要为所有部分运行这个 SP,所以我的想法是在游标内执行 SP,并继续传递下一部分,直到没有剩余部分需要处理。我不知道如何做到这一点,或者是否可能,因为在游标的 SELECT 语句中(我在其中使用存储过程代码)我需要传递一个变量才能使其工作。

这是存储过程的代码:

CREATE PROCEDURE dbo.usp_BuildBOMLit
@item_no CHAR(8) = NULL 
AS 
BEGIN
WITH CTE AS (
        SELECT DISTINCT
            LTRIM(RTRIM(lvl1.item_no)) as item_no, LTRIM(RTRIM(lvl1.comp_item_no)) as comp_item_no,
            CASE
                WHEN LTRIM(RTRIM(lvl1.comp_item_no)) LIKE '.%' 
                    THEN 
                        LTRIM(RTRIM(lvl1.comp_item_no))
                WHEN LTRIM(RTRIM(lvl1.comp_item_no)) LIKE 'E%' 
                    THEN
                        (SELECT TOP 1 LTRIM(RTRIM(comp_item_no)) FROM bmprdstr_sql WHERE LTRIM(RTRIM(item_no))=LTRIM(RTRIM(lvl1.comp_item_no))  AND LTRIM(RTRIM(comp_item_no)) LIKE '.%')
                WHEN LTRIM(RTRIM(lvl1.comp_item_no)) LIKE 'ZG%'
                    THEN 
                        (SELECT TOP 1 LTRIM(RTRIM(comp_item_no)) FROM bmprdstr_sql WHERE LTRIM(RTRIM(item_no))=LTRIM(RTRIM(lvl1.comp_item_no))  AND LTRIM(RTRIM(comp_item_no)) LIKE '.%')
                    ELSE
                        NULL
            END as lvl_2_comp_item_no
        FROM 
            bmprdstr_sql as lvl1 
            LEFT JOIN bmprdstr_sql lvl2 ON lvl1.comp_item_no=lvl2.item_no
        WHERE 
            (lvl1.item_no = @item_no)
            AND (lvl1.comp_item_no LIKE '.%' OR lvl1.comp_item_no LIKE 'ZG%' OR lvl1.comp_item_no LIKE 'E%')
    )
    SELECT DISTINCT
        CASE
            WHEN LEFT(item_no,1)='.'
                THEN STUFF(item_no,1,1,'')
            ELSE
                item_no
        END as item_no,
        part_no = 
            STUFF((SELECT DISTINCT ',' + 
                    CASE
                        WHEN LEFT(lvl_2_comp_item_no,1)='.'
                            THEN STUFF(lvl_2_comp_item_no,1,1,'')
                        ELSE lvl_2_comp_item_no
                        END
                    FROM CTE where item_no=@item_no FOR XML PATH('')),1,1,'')
    FROM
        CTE
    WHERE
        lvl_2_comp_item_no IS NOT NULL AND item_no IS NOT NULL
END

输出正是我需要的格式:

项目编号 |零件号

JM9027 | GS10702,LB2391,LB2704,LB2834,LB2896,LB6996

当我创建游标时,我在游标的

SELECT
语句中使用相同的代码,但是正如您所看到的,是否需要传入父级 (
@itemno
)。我已经尝试过此操作,但无济于事:

 SET NOCOUNT ON;
 DECLARE @itemno CHAR(15);
 DECLARE @partno VARCHAR(254);
 DECLARE @outside_cursor AS CURSOR;

SET @outside_cursor = CURSOR FAST_FORWARD FOR 
WITH CTE AS
(SELECT DISTINCT 
        LTRIM(RTRIM(lvl1.item_no)) AS item_no,
        LTRIM(RTRIM(lvl1.comp_item_no)) AS comp_item_no,
        CASE
            WHEN LTRIM(RTRIM(lvl1.comp_item_no)) LIKE '.%' 
                THEN LTRIM(RTRIM(lvl1.comp_item_no))
            WHEN LTRIM(RTRIM(lvl1.comp_item_no)) LIKE 'E%' 
                THEN
                    (SELECT TOP 1 
                        LTRIM(RTRIM(comp_item_no))
                    FROM 
                        bmprdstr_sql
                    WHERE 

LTRIM(RTRIM(item_no))=LTRIM(RTRIM(lvl1.comp_item_no))
                        AND LTRIM(RTRIM(comp_item_no)) LIKE '.%'
                    )
            WHEN LTRIM(RTRIM(lvl1.comp_item_no)) LIKE 'ZG%' 
                THEN
                    (SELECT TOP 1 
                        LTRIM(RTRIM(comp_item_no))
                    FROM 
                        bmprdstr_sql
                    WHERE 

LTRIM(RTRIM(item_no))=LTRIM(RTRIM(lvl1.comp_item_no))
                        AND LTRIM(RTRIM(comp_item_no)) LIKE '.%'
                    )
            ELSE NULL
        END AS lvl_2_comp_item_no
 FROM 
    bmprdstr_sql AS lvl1
    LEFT JOIN bmprdstr_sql lvl2 ON lvl1.comp_item_no=lvl2.item_no
 WHERE 
    (lvl1.item_no = @itemno) -- <-- problem
     AND (lvl1.comp_item_no LIKE '.%' OR lvl1.comp_item_no LIKE 'ZG%' 
OR lvl1.comp_item_no LIKE 'E%')
)
SELECT DISTINCT 
    CASE
        WHEN LEFT(item_no,1)='.' 
            THEN STUFF(item_no,1,1,'')
        ELSE item_no
    END AS item_no,
    part_no = 
        STUFF(
            (SELECT DISTINCT ',' + 
                CASE 
                    WHEN LEFT(lvl_2_comp_item_no,1)='.' 
                        THEN STUFF(lvl_2_comp_item_no,1,1,'') 
                    ELSE lvl_2_comp_item_no 
                END
            FROM 
                CTE
            WHERE 
                item_no=@itemno FOR XML PATH('')),1,1,'') 
                     -- ^ problem
FROM 
    CTE
WHERE 
    lvl_2_comp_item_no IS NOT NULL
    AND item_no IS NOT NULL 

OPEN @outside_cursor;
FETCH NEXT FROM @outside_cursor INTO @itemno, @partno;

WHILE @@FETCH_STATUS = 0 
    BEGIN
        INSERT INTO items_parts (item_no, part_no)
        VALUES (@itemno, @partno) 
    FETCH NEXT FROM @outside_cursor INTO @itemno,  @partno 
    END 
CLOSE @outside_cursor 
DEALLOCATE @outside_cursor

关于如何实现这一点有什么建议吗?

sql-server sql-server-2008 t-sql stored-procedures cursor
3个回答
0
投票

不管您是否需要光标,并将您的问题视为学术问题,您都以错误的方式使用光标。

你有一个进程可以做你想做的事,但仅限于一个父母。 合并游标的方法是使用游标遍历父级,并在每次循环时简单地用新的 item_no 填充 item_no 变量,然后在游标内运行现有代码。

您不需要在两个单独的查询中执行 SELECT,然后执行 INSERT,您可以在游标中执行单个 INSERT..SELECT。 光标只是遍历 item_no 值列表。


0
投票

我明白了;

我认为你拥有的是一个 CTE,它返回一行数据(或者可能是 (n) 行数据,但仅对 @itemno 是唯一的)

    CTE(item_no, comp_item_no, lvl_2_comp_item_no) 
      AS (
            --.... and the row will correspond to the @itemno provided

但你想要的是 CTE,它返回所有可能的 [ietm_no] 的值集(这应该会在幕后产生一个临时表,其中包含 3 列 [item_no]、[comp_item_no]、[lvl_2_comp_item_no] 和 (n )行。

然后您可以使用游标逐一循环这些,将它们加载到游标变量中,即:

FETCH NEXT FROM @outside_cursor INTO @itemno, @partno...

并使用唯一的@itemno 处理每行的插入。

顺便说一句,我注意到您将光标命名为@outside_cursor,这意味着某处也可能有一个@inside_cursor?如果您嵌套游标(这不是最佳选择),请注意 @@FETCH_STATUS 是一个全局变量,您需要为每个游标单独管理此变量中的值。

我没看错吗?


0
投票
我能够使用存储过程和自连接查询来完成我需要的任务 - 不需要游标。

© www.soinside.com 2019 - 2024. All rights reserved.