Database/MSSQL work
PLAN CACHE에서 단일 실행계획을 찾고, 또 삭제하는 캐시 쿼리 명령어
웨이칭
2020. 11. 25. 10:53
-- 현재 캐시된것들 확인하기
SELECT
*
FROM
SYS.dm_exec_cached_plans
GO
-- PLAN CACHE에서 단일 실행계획을 찾는 쿼리문(PLAN CACHE사이즈를 최적화하기위함)
아래 출력되는 것들을 재 컴파일하면 속도를 줄일 수 있음
--> 한번만 쓰고 버리는 쿼리들 위주로 재컴파일 필요,,,,, 해당 쿼리문은 재활용 하자!!!!
SELECT
TEXT, CP.OBJTYPE, CP.SIZE_IN_BYTES
FROM
SYS.dm_exec_cached_plans AS cp
CROSS APPLY
SYS.dm_exec_sql_text(CP.PLAN_HANDLE) ST
WHERE
CP.cacheobjtype = 'Compiled Plan'
AND
-- Adhoc은 쿼리를 한번 던지는것, Prepared은 사전에 필요한 것들을 의미
CP.OBJTYPE IN('Adhoc','Prepared')
AND
CP.USECOUNTS = 1 --1번만 실행하고 아직 CACHE에 있다.
ORDER BY
CP.SIZE_IN_BYTES DESC
OPTION(RECOMPILE)
GO
아래와 같이 출력됨
--한번만 실행된 쿼리문은 저장이 안되게(조금만 차지하게) 활성화시킬 수 있다.
PLAN CACHE에서 단일 사용계획에 사용되는 메모리양을 확인 할 때 사용하는 쿼리
SELECT
objtype AS [CacheType]
, count_big(*) AS [Total Plans]
, sum(cast(size_in_bytes as decimal(18,2)))/1024/1024 AS [Total MBs]
, avg(usecounts) AS [Avg Use Count]
, sum(cast((CASE WHEN usecounts = 1 THEN size_in_bytes ELSE 0 END) as decimal(18,2)))/1024/1024 AS [Total MBs - USE Count 1]
, sum(CASE WHEN usecounts = 1 THEN 1 ELSE 0 END) AS [Total Plans - USE Count 1]
FROM sys.dm_exec_cached_plans
GROUP BY objtype
ORDER BY [Total MBs - USE Count 1] DESC
-- 한번 썼던 PLAN CACHE 쿼리들을 삭제하는 명령어
DECLARE @MB decimal(19,3) , @Count bigint, @StrMB nvarchar(20)
SELECT
@MB = sum(cast((CASE WHEN usecounts = 1 AND objtype
IN ('Adhoc', 'Prepared') THEN -- 범위를 정해줌
size_in_bytes ELSE 0 END)
as decimal(12,2)))/1024/1024
,@Count = sum(CASE WHEN usecounts = 1 AND objtype
IN ('Adhoc', 'Prepared') THEN 1 ELSE 0 END)
,@StrMB = convert(nvarchar(20), @MB)
FROM sys.dm_exec_cached_plans
IF @MB > 10
BEGIN
DBCC FREESYSTEMCACHE('SQL Plans') --10보다 크면 DBCC명령문으로 지워줌
RAISERROR ('%s MB was allocated to single-use plan cache. Single-use plans have been cleared.', 10, 1, @StrMB)
END
ELSE
BEGIN
RAISERROR ('Only %s MB is allocated to single-use plan cache – no need to clear cache now.', 10, 1, @StrMB)
END
GO
코드를 실행하면 아래와같이 메세지가 뜬다.
그리고 다시 PLAN CACHE 단일실행 계획을 찾는 쿼리문을 실행해본다.
그럼 결과가 매우 많이 줄어든것을 알 수 있다.
최적화가 필요한 쿼리 찾기-- 건너뛰어도 됨......
/*
최적화가 필요한 PlanCache 확인(찾기)
인덱스가 없는 현재 모든 클랜 캐시를 찾음
*/
SELECT
dec.usecounts,
dec.refcounts,
dec.objtype,
dec.cacheobjtype,
des.dbid,
des.text,
deq.query_plan
FROM
sys.dm_exec_cached_plans AS dec
CROSS APPLY
sys.dm_exec_sql_text(dec.plan_handle) AS des
CROSS APPLY
sys.dm_exec_query_plan(dec.plan_handle) AS deq
WHERE
deq.query_plan.exist
(N'/ShowPlanXML/BatchSequence/Batch/Statements/StmtSimple/QueryPlan/MissingIndexes/MissingIndexGroup') <> 0
ORDER BY dec.usecounts DESC
go
--인덱스가 있는것.
WITH XMLNAMESPACES(DEFAULT N'http://schemas.microsoft.com/sqlserver/2004/07/showplan')
SELECT
cp.query_hash,
cp.query_plan_hash,
ConvertIssue= operators.value('@ConvertIssue', 'nvarchar(250)'),
Expression= operators.value('@Expression', 'nvarchar(250)'),
qp.query_plan
FROM
sys.dm_exec_query_stats cp
CROSS APPLY
sys.dm_exec_query_plan(cp.plan_handle) qp
CROSS APPLY
query_plan.nodes('//Warnings/PlanAffectingConvert') rel(operators) --xml로 나옴
go
그리고 다시 PLAN CACHE 단일사용계획에 사용되는 메모리양을 확인하는 쿼리를 사용해본다.
이또한 엄청 줄어든것을 알 수 있다.