SQL Server Table Hints بهینه سازی کوئری و بهبود عملکرد

بهینه‌سازی کوئری‌ها در SQL Server: راهنمای کامل Table Hints برای عملکرد بهتر

SQL Server Table Hints ابزارهایی قدرتمند برای کنترل رفتار بهینه‌ساز کوئری هستند که به شما امکان می‌دهند نحوه دسترسی SQL Server به جداول را به‌طور دستی مدیریت کنید. این قابلیت که در SQL Server 2000 و نسخه‌های بعدی موجود است، به متخصصان دیتابیس کمک می‌کند تا در شرایط خاص بر بهینه‌ساز غلبه کرده و عملکرد یا همزمانی (Concurrency) را بهبود بخشند. با این حال، استفاده از این Hints باید با دقت و آگاهی کامل انجام شود، زیرا به‌طور کلی بهینه‌ساز SQL Server در انتخاب بهترین طرح اجرایی (Execution Plan) بسیار کارآمد عمل می‌کند.

استفاده از Table Hints معمولاً تنها زمانی توصیه می‌شود که با یک مشکل عملکردی مشخص روبرو هستید که بهینه‌ساز قادر به حل آن نیست، یا نیاز به مدیریت خاص قفل‌ها برای جلوگیری از بن‌بست (Deadlock) دارید. Hint ها در بند `FROM` یک دستور `SELECT`, `INSERT`, `UPDATE`, یا `DELETE` و با استفاده از کلمه کلیدی `WITH` مشخص می‌شوند.

    
        FROM <table_source> WITH ( <table_hint> [ ,...n ] )
    

در ادامه به بررسی برخی از پرکاربردترین SQL Server Table Hints و کاربردهای آن‌ها می‌پردازیم:

NOLOCK و READUNCOMMITTED

این Hints یکی از رایج‌ترین گزینه‌ها برای بهبود عملکرد خواندن (Read) در دیتابیس‌های پرکار هستند.
NOLOCK و READUNCOMMITTED مشخص می‌کنند که هیچ قفل اشتراکی صادر نشود و هیچ قفل انحصاری (Exclusive) اعمال نگردد. با استفاده از این SQL Server Hint، امکان خواندن تراکنش‌های commit نشده وجود دارد که می‌تواند منجر به خطای Dirty Read شود.

(NOLOCK)

(READUNCOMMITTED)

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

HOLDLOCK و SERIALIZABLE

این Hints به SQL Server دستور می‌دهند که قفل‌های اشتراکی را تا پایان تراکنش حفظ کند. این کار سطح بالایی از انزوا (Isolation) را فراهم می‌کند و از تغییر داده‌ها توسط سایر تراکنش‌ها جلوگیری می‌نماید.
HOLDLOCK قفل‌های اشتراکی را تا پایان تراکنش نگه می‌دارد. این Hint معادل سطح انزوای SERIALIZABLE است.

(HOLDLOCK)

SERIALIZABLE یک قفل محدوده (Range Lock) بر روی مجموعه داده‌ها قرار می‌دهد و از به‌روزرسانی یا درج سطرها در آن محدوده توسط سایر کاربران تا زمان اتمام تراکنش جلوگیری می‌کند. این Hint نیز معادل HOLDLOCK است.

(SERIALIZABLE)

استفاده از این Hint‌ها برای تضمین ثبات داده‌ها در محیط‌هایی که نیاز به انزوای بالا دارند، مفید است، اما می‌تواند رقابت (Contention) را افزایش دهد.

REPEATABLEREAD و READCOMMITTED

REPEATABLEREAD قفل‌هایی را بر روی تمام داده‌های استفاده شده در یک کوئری قرار می‌دهد و از به‌روزرسانی آن داده‌ها توسط کاربران دیگر جلوگیری می‌کند، اما اجازه درج رکوردهای جدید (Phantom Reads) را می‌دهد.

(REPEATABLEREAD)

READCOMMITTED مشخص می‌کند که قفل‌های اشتراکی هنگام خواندن داده‌ها نگه داشته می‌شوند، اما پس از اتمام عملیات خواندن آزاد می‌شوند. این رفتار، تنظیم پیش‌فرض در SQL Server است.

(READCOMMITTED)

این Hint برای سناریوهایی مناسب است که نیاز به خواندن چندین بار یک مجموعه داده در یک تراکنش وجود دارد و می‌خواهید از تغییر آن توسط دیگران جلوگیری کنید، اما نیاز به انزوای کامل (مثل SERIALIZABLE) ندارید.

UPDLOCK و XLOCK

UPDLOCK مشخص می‌کند که قفل‌های به‌روزرسانی (Update Locks) گرفته شده و تا پایان تراکنش نگه داشته شوند. این Hint از گرفتن قفل‌های به‌روزرسانی یا انحصاری توسط سایر فرآیندها جلوگیری می‌کند و برای جلوگیری از بن‌بست در عملیات به‌روزرسانی مفید است.

(UPDLOCK)

XLOCK مشخص می‌کند که قفل‌های انحصاری (Exclusive Locks) گرفته شده و تا پایان تراکنش نگه داشته شوند. این Hint از گرفتن قفل‌های اشتراکی یا انحصاری توسط سایر فرآیندها جلوگیری می‌کند.

(XLOCK)

این Hint‌ها برای تضمین انحصار در عملیات تغییر داده‌ها یا زمانی که می‌خواهید از دسترسی همزمان به یک منبع خاص جلوگیری کنید، کاربردی هستند.

ROWLOCK، PAGLOCK و TABLOCK

این Hint‌ها به SQL Server دستور می‌دهند که از چه سطح از قفل‌گذاری (Locking Granularity) استفاده کند.
ROWLOCK مشخص می‌کند که هنگام قفل‌گذاری از قفل‌های در سطح سطر (Row-level Locks) استفاده شود.

(ROWLOCK)

PAGLOCK مشخص می‌کند که هنگام قفل‌گذاری از قفل‌های در سطح صفحه (Page-level Locks) استفاده شود.

(PAGLOCK)

TABLOCK مشخص می‌کند که یک قفل در سطح جدول (Table-level Lock) گرفته شود. این می‌تواند با کاهش سربار قفل‌گذاری، عملکرد را بهبود بخشد، اما همزمانی را نیز کاهش می‌دهد.

(TABLOCK)

TABLOCKX یک قفل انحصاری بر روی جدول می‌گیرد.

(TABLOCKX)

انتخاب سطح قفل‌گذاری مناسب بین عملکرد و همزمانی تعادل ایجاد می‌کند. ROWLOCK بیشترین همزمانی و TABLOCK کمترین همزمانی را فراهم می‌کند.

READPAST

READPAST برای سناریوهای صف (Queue Processing) بسیار مفید است. این Hint سطرها و صفحات قفل شده توسط تراکنش‌های دیگر را نادیده می‌گیرد و از آن‌ها عبور می‌کند.

(READPAST)

این Hint به کوئری‌ها اجازه می‌دهد تا بدون انتظار برای آزاد شدن قفل‌ها، به پردازش ادامه دهند و داده‌های در دسترس را بخوانند.

INDEX

این Hint به بهینه‌ساز کوئری دستور می‌دهد که از یک یا چند ایندکس مشخص برای دسترسی به داده‌ها استفاده کند.

(INDEX (index_val [,…n]))

با استفاده از (INDEX (0)) می‌توانید از Table Scan جلوگیری کنید. این Hint می‌تواند برای بهینه‌سازی دسترسی به داده‌ها در کوئری‌های پیچیده مفید باشد و زمانی استفاده می‌شود که بهینه‌ساز انتخاب نادرستی برای ایندکس داشته باشد.

FORCESEEK

FORCESEEK به بهینه‌ساز کوئری دستور می‌دهد که حتماً از یک عملیات Seek بر روی ایندکس مشخص شده استفاده کند. این Hint برای زمانی است که می‌دانید Seek کارآمدتر از Scan است.

(FORCESEEK)

FAST

این Hint به بهینه‌ساز دستور می‌دهد که بازیابی `number_rows` را به سرعت انجام دهد. این برای زمانی مفید است که شما فقط به چند ردیف اول یک مجموعه نتایج بزرگ نیاز دارید.

(FAST (number_rows))

MAXDOP

MAXDOP (Maximum Degree of Parallelism) مقدار حداکثر درجه موازی‌سازی را برای یک کوئری خاص override می‌کند.

(MAXDOP (number))

با این Hint می‌توانید میزان استفاده از پردازنده‌های چندگانه را برای اجرای یک کوئری مشخص کنترل کنید.

NOEXPAND

NOEXPAND به بهینه‌ساز کوئری دستور می‌دهد که از گسترش (Expanding) نماهای ایندکس شده (Indexed Views) جلوگیری کند.

(NOEXPAND)

این Hint زمانی مفید است که می‌خواهید مطمئن شوید که بهینه‌ساز از نمای ایندکس شده به همان شکلی که تعریف شده است، استفاده می‌کند.

KEEPIDENTITY و KEEPDEFAULTS

این Hints در دستورات `INSERT` کاربرد دارند.
KEEPIDENTITY به شما اجازه می‌دهد تا مقادیر Identity را به صراحت درج کنید.

(KEEPIDENTITY)

KEEPDEFAULTS اجازه می‌دهد تا مقادیر Null به جای مقادیر پیش‌فرض (Default Values) درج شوند.

(KEEPDEFAULTS)

این Hint‌ها کنترل دقیقی بر نحوه درج داده‌ها در جداول با ستون‌های Identity یا Default فراهم می‌کنند.

نمونه‌ای از کاربرد Table Hint

    
        SELECT au_lname, au_fname FROM authors WITH (NOLOCK);
    

در این مثال، `(NOLOCK)` به SQL Server دستور می‌دهد که بدون اعمال قفل‌های اشتراکی بر روی جدول `authors`، داده‌ها را بخواند.

نتیجه‌گیری

Table Hints در SQL Server ابزارهای قدرتمندی هستند که می‌توانند برای عیب‌یابی و بهینه‌سازی عملکرد کوئری‌ها و مدیریت همزمانی استفاده شوند. با این حال، استفاده بی‌رویه یا نادرست از آن‌ها می‌تواند منجر به مشکلات پیش‌بینی نشده‌ای مانند Dirty Reads یا کاهش همزمانی شود. توصیه می‌شود قبل از به‌کارگیری هر Hint، تأثیر آن بر طرح اجرایی و رفتار سیستم به‌طور کامل بررسی و آزمایش شود تا از عملکرد بهینه و ثبات دیتابیس اطمینان حاصل شود. در بیشتر موارد، اجازه دادن به بهینه‌ساز کوئری برای انجام کارش بهترین است، اما دانستن نحوه استفاده صحیح از Hint‌ها می‌تواند در شرایط خاص نجات‌بخش باشد.

 

sql serverTable Hints
Comments (0)
Add Comment