شناسایی رفع و پیشگیری Deadlock در SQL Server

شناسایی و رفع Deadlock در SQL Server: راهنمای جامع و عملی

Deadlock یکی از چالش‌های رایج در سیستم‌های مدیریت پایگاه داده، به ویژه SQL Server، است که می‌تواند عملکرد سیستم را به شدت تحت تأثیر قرار دهد و تجربه کاربری نامطلوبی ایجاد کند. این وضعیت زمانی رخ می‌دهد که دو یا چند فرآیند (تراکنش) به صورت همزمان منتظر منابعی هستند که توسط یکدیگر قفل شده‌اند. در این مقاله به بررسی عمیق چگونگی وقوع Deadlock، نحوه شناسایی آن و راهکارهای مؤثر برای پیشگیری و رفع این مشکل خواهیم پرداخت.

Deadlock چیست و چگونه اتفاق می‌افتد؟

Deadlock وضعیتی است که در آن دو یا چند تراکنش برای دسترسی به منابعی که توسط تراکنش‌های دیگر درگیر در بن‌بست قفل شده‌اند، به طور ابدی منتظر می‌مانند. SQL Server برای حفظ یکپارچگی داده‌ها از مکانیزم قفل‌گذاری استفاده می‌کند. با این حال، استفاده نادرست یا رقابت بالا برای منابع مشترک می‌تواند به Deadlock منجر شود.

سناریوی کلاسیک Deadlock به شرح زیر است:
1. تراکنش A منبع X را قفل می‌کند و سپس سعی می‌کند منبع Y را قفل کند.
2. در همین زمان، تراکنش B منبع Y را قفل می‌کند و سپس سعی می‌کند منبع X را قفل کند.
نتیجه این می‌شود که تراکنش A منتظر آزاد شدن Y توسط B است، در حالی که B منتظر آزاد شدن X توسط A است. هیچ‌کدام نمی‌توانند ادامه دهند و هر دو تراکنش برای همیشه منتظر می‌مانند. SQL Server به طور خودکار یکی از تراکنش‌ها را به عنوان “قربانی” انتخاب کرده و آن را خاتمه می‌دهد (rollback می‌کند) تا تراکنش دیگر بتواند ادامه یابد. این فرآیند انتخاب قربانی Deadlock Victim نامیده می‌شود.

شناسایی Deadlock در SQL Server

شناسایی Deadlock برای حل مشکلات عملکردی حیاتی است. SQL Server ابزارهای متعددی را برای تشخیص و تحلیل Deadlock ارائه می‌دهد.

۱. استفاده از نمایش‌های مدیریت پویا (DMVs)

می‌توانید از DMVهایی مانند `sys.dm_exec_requests` و `sys.dm_os_waiting_tasks` برای مشاهده وضعیت قفل‌ها و تراکنش‌های در انتظار استفاده کنید. اگرچه این DMVها به طور مستقیم Deadlock را نشان نمی‌دهند، اما می‌توانند نشانه‌هایی از آن را ارائه دهند.


SELECT
    request_session_id,
    resource_type,
    resource_database_id,
    resource_associated_entity_id,
    request_mode,
    request_status,
    request_owner_type,
    request_type,
    request_start_time,
    blocking_session_id
FROM
    sys.dm_exec_requests
WHERE
    blocking_session_id <> 0;

این کوئری تمام درخواست‌هایی را نشان می‌دهد که توسط sessionهای دیگر بلاک شده‌اند. مشاهده زنجیره‌های طولانی بلاک شدن می‌تواند نشانه‌ای از مشکلات قفل‌گذاری باشد که ممکن است منجر به Deadlock شود.


SELECT
    waiting_task_address,
    wait_duration_ms,
    wait_type,
    resource_address,
    blocking_task_address,
    session_id,
    resource_description
FROM
    sys.dm_os_waiting_tasks
WHERE
    wait_type LIKE 'LCK_%';

این کوئری وظایفی را نشان می‌دهد که منتظر قفل هستند و جزئیات مربوط به منبع و وظیفه بلاک‌کننده را ارائه می‌دهد.

۲. Extended Events

Extended Events قدرتمندترین ابزار برای ردیابی و تحلیل Deadlock در SQL Server است. شما می‌توانید یک Extended Event Session را برای رصد رویداد `deadlock_report` پیکربندی کنید. این رویداد یک گراف XML شامل تمام جزئیات مربوط به Deadlock، از جمله تراکنش‌های درگیر، منابع قفل شده، و تراکنش قربانی را فراهم می‌کند.

برای ایجاد یک Extended Event Session برای ردیابی Deadlock:


CREATE EVENT SESSION [DeadlockMonitor]
ON SERVER
    ADD EVENT sqlserver.deadlock_report
    ADD TARGET package0.asynchronous_file_target
    (SET filename = N'C:\Temp\DeadlockMonitor.xel', max_file_size = (50), max_rollover_files = (5))
WITH
    (MAX_MEMORY = 4096 KB, EVENT_RETENTION_MODE = ALLOW_SINGLE_EVENT_LOSS, MAX_DISPATCH_LATENCY = 30 SECONDS, MAX_EVENT_SIZE = 0 KB, MEMORY_PARTITION_MODE = NONE,
     TRACK_CAUSALITY = OFF, STARTUP_STATE = OFF);
GO

ALTER EVENT SESSION [DeadlockMonitor]
ON SERVER STATE = START;
GO

پس از وقوع Deadlock، می‌توانید فایل `.xel` را با SQL Server Management Studio باز کنید تا گراف Deadlock را مشاهده کنید. این گراف یک نمایش بصری از چرخه Deadlock ارائه می‌دهد و به شما کمک می‌کند تا به راحتی منابع و تراکنش‌های درگیر را شناسایی کنید.

شبیه‌سازی یک Deadlock

برای درک بهتر، بیایید یک سناریوی Deadlock ساده را شبیه‌سازی کنیم. ابتدا دو جدول موقت ایجاد می‌کنیم:


CREATE TABLE #TableA (ID INT PRIMARY KEY, ValueA VARCHAR(50));
CREATE TABLE #TableB (ID INT PRIMARY KEY, ValueB VARCHAR(50));

INSERT INTO #TableA VALUES (1, 'Data A1');
INSERT INTO #TableB VALUES (1, 'Data B1');

حالا دو پنجره کوئری در SQL Server Management Studio باز کنید و کوئری‌های زیر را به ترتیب اجرا کنید:

پنجره ۱ (تراکنش A):


BEGIN TRAN;
UPDATE #TableA SET ValueA = 'Update A' WHERE ID = 1;
WAITFOR DELAY '00:00:05'; -- منتظر می‌ماند تا تراکنش B شروع به کار کند
UPDATE #TableB SET ValueB = 'Update B from A' WHERE ID = 1;
COMMIT TRAN;

پنجره ۲ (تراکنش B):


BEGIN TRAN;
UPDATE #TableB SET ValueB = 'Update B' WHERE ID = 1;
WAITFOR DELAY '00:00:05'; -- منتظر می‌ماند تا تراکنش A به مرحله دوم برسد
UPDATE #TableA SET ValueA = 'Update A from B' WHERE ID = 1;
COMMIT TRAN;

ابتدا کوئری پنجره ۱ را اجرا کنید، سپس بلافاصله کوئری پنجره ۲ را اجرا کنید. پس از چند ثانیه، یکی از تراکنش‌ها با خطای Deadlock خاتمه می‌یابد و دیگری با موفقیت اجرا می‌شود.

پیشگیری و رفع Deadlock

پیشگیری از Deadlock بهتر از درمان آن است. در اینجا چند استراتژی کلیدی برای کاهش احتمال وقوع Deadlock و رفع آن آورده شده است:

۱. کوتاه و سریع نگه داشتن تراکنش‌ها

هرچه تراکنش‌ها کوتاه‌تر باشند و منابع را برای زمان کمتری قفل کنند، احتمال وقوع Deadlock کاهش می‌یابد. از نگه داشتن تراکنش‌های باز در حین انتظار برای ورودی کاربر یا پردازش‌های طولانی خارج از پایگاه داده خودداری کنید.

۲. دسترسی به اشیاء به ترتیب ثابت

همیشه سعی کنید به اشیاء (جداول، ردیف‌ها) با ترتیب ثابت دسترسی پیدا کنید. در مثال بالا، اگر هر دو تراکنش ابتدا #TableA و سپس #TableB را قفل می‌کردند، Deadlock رخ نمی‌داد.

۳. استفاده از ایندکس‌های مناسب

ایندکس‌های کارآمد می‌توانند به کاهش مدت زمان نگهداری قفل‌ها کمک کنند، زیرا تراکنش‌ها می‌توانند سریع‌تر به داده‌های مورد نیاز خود دسترسی پیدا کنند و قفل‌های کمتری را درگیر کنند.

۴. کاهش سطح Isolation تراکنش (Transaction Isolation Level)

تنظیم سطح Isolation پایین‌تر مانند `READ COMMITTED` (که پیش‌فرض است) یا `READ UNCOMMITTED` می‌تواند از Deadlock جلوگیری کند، اما ممکن است به قیمت سازگاری داده‌ها تمام شود. با این حال، استفاده از `SNAPSHOT ISOLATION` یا `READ COMMITTED SNAPSHOT ISOLATION` می‌تواند به طور مؤثری Deadlock را کاهش دهد، زیرا از قفل‌های خواندن جلوگیری می‌کند.


ALTER DATABASE YourDatabaseName SET READ_COMMITTED_SNAPSHOT ON;

۵. استفاده از Lock Hints ها با احتیاط

استفاده از Lock Hints مانند `NOLOCK` یا `ROWLOCK` می‌تواند به کنترل رفتار قفل‌گذاری کمک کند، اما باید با دقت و درک کامل عواقب آن انجام شود. استفاده نادرست می‌تواند منجر به خواندن داده‌های ناهمگون یا مشکلات دیگر شود.


SELECT * FROM YourTable WITH (NOLOCK) WHERE ID = 1;

۶. تنظیم اولویت Deadlock (DEADLOCK_PRIORITY)

می‌توانید اولویت یک تراکنش را در هنگام وقوع Deadlock تنظیم کنید. تراکنشی با اولویت بالاتر، قربانی Deadlock نخواهد شد، در حالی که تراکنشی با اولویت پایین‌تر ممکن است انتخاب شود.


SET DEADLOCK_PRIORITY HIGH;

یا


SET DEADLOCK_PRIORITY LOW;

این گزینه به شما امکان می‌دهد کنترل کنید که کدام تراکنش در صورت وقوع Deadlock خاتمه یابد و کدام ادامه دهد.

نتیجه‌گیری

Deadlock در SQL Server یک مشکل پیچیده است که نیاز به درک عمیق از مکانیزم‌های قفل‌گذاری و رفتار تراکنش‌ها دارد. با استفاده از ابزارهای قدرتمند SQL Server مانند Extended Events و DMVs برای شناسایی دقیق علت، و پیاده‌سازی استراتژی‌های پیشگیرانه مانند کوتاه نگه داشتن تراکنش‌ها، دسترسی مرتب به منابع، و استفاده از سطوح Isolation مناسب، می‌توانید به طور مؤثری از وقوع Deadlock جلوگیری کرده و عملکرد و پایداری سیستم پایگاه داده خود را بهبود بخشید. با این رویکرد جامع، توسعه‌دهندگان می‌توانند از تجربه کاربری بی‌نقصی اطمینان حاصل کنند.

 

من علی دستجردی‌ام؛ عاشق کار با دیتا، از SQL Server تا بیگ‌دیتا و هوش مصنوعی. دغدغه‌ام کشف ارزش داده‌ها و به‌اشتراک‌گذاری تجربه‌هاست. ✦ رزومه من: alidastjerdi.com ✦

عضویت
منو باخبر کن!!!
نام
ایمیل

0 دیدگاه
Inline Feedbacks
دیدن تمامی کامنتها

فوتر سایت

ورود به سایت

sqlyar

هنوز عضو نیستید؟

ورود به سایت

هنوز تبت نام نکردید ؟