Как я могу избежать функции 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)