Как я могу избежать функции USER_ID() в предложении WHERE

Я пытаюсь оптимизировать запрос следующим образом:

SELECT  id, type, parent_id
FROM    [dbo].[v5] with(nolock)
WHERE 
(
    USER_ID() = 1
    OR
        (
            v5.type IN (1,2,3)
            and 
            EXISTS
            (
                    SELECT obj_id FROM dbo.access with(nolock) 
                    where  access.obj_id = id 
                    AND obj_id.[u_name] in 
                    (
                        select SUSER_SNAME() union all 
                        select [name] COLLATE database_default from dbo.u_role
                    )
            )
        )
)

Работает 25 сек. Когда я комментирую эти строки:

--USER_ID() = 1
 --     OR

это работает 5 сек. Как оптимизировать этот запрос? Как избежать использования USER_ID() в предложении where?

1 ответ

Попробуй это -

Вариант № 1:

DECLARE @user INT
SELECT @user = USER_ID() --<--

SELECT 
      v.id
    , v.[type]
    , v.parent_id
FROM dbo.v5 v WITH (NOLOCK)
LEFT JOIN (
    SELECT DISTINCT a.obj_id 
    FROM dbo.access a WITH(NOLOCK) 
    WHERE a.[u_name] IN 
        (
            SELECT SUSER_SNAME() 
            UNION ALL 
            SELECT [name] COLLATE database_default 
            FROM dbo.u_role
        )
) a ON a.obj_id = v.id 
WHERE @user = 1
    OR (
            v5.[type] IN (1, 2, 3) 
        AND 
            a.obj_id IS NOT NULL
    )

Вариант № 2:

IF USER_ID() = 1 BEGIN

    SELECT 
          v.id
        , v.[type]
        , v.parent_id
    FROM dbo.v5 v WITH (NOLOCK)

END
ELSE BEGIN

    SELECT 
          v.id
        , v.[type]
        , v.parent_id
    FROM dbo.v5 v WITH (NOLOCK)
    JOIN (
        SELECT DISTINCT a.obj_id 
        FROM dbo.access a WITH(NOLOCK) 
        WHERE a.[u_name] IN 
            (
                SELECT SUSER_SNAME() 
                UNION ALL 
                SELECT [name] COLLATE database_default 
                FROM dbo.u_role
            )
    ) a ON a.obj_id = v.id 
    WHERE v5.[type] IN (1, 2, 3) 

END

Вариант № 3:

DECLARE @user INT
SELECT @user = USER_ID()

SELECT  id, [type], parent_id
FROM [dbo].[v5] WITH(NOLOCK)
WHERE @user = 1
    OR (
        v5.[type] IN (1,2,3)
        AND 
        EXISTS (
            SELECT 1 
            FROM dbo.access WITH(NOLOCK) 
            WHERE access.obj_id = id 
                AND access.[u_name] IN 
                    (
                        SELECT SUSER_SNAME() 
                        UNION ALL 
                        SELECT [name] COLLATE database_default 
                        FROM dbo.u_role
                    )
        )
    )
OPTION (RECOMPILE)
Другие вопросы по тегам