資料表提示 (Transact-SQL)
適用於:Microsoft Fabric 中的 SQL ServerAzure SQL 資料庫 Azure SQL 受控執行個體SQL 資料庫
資料表提示可用來在資料操作語言 (DML) 陳述式期間覆寫查詢最佳化工具的預設行為。 您可以指定鎖定方法、一或多個索引、查詢處理作業 (例如資料表掃描或索引搜尋),或是其他選項。 數據表提示是在 DML 語句的 FROM
子句中指定,而且只會影響該子句中所參考的數據表或檢視表。
警告
由於 SQL Server 查詢最佳化工具通常會選取最好的查詢執行計畫,因此我們建議只有資深的開發人員和資料庫管理員才應該使用提示,並將其當作最後的解決辦法。
適用範圍:
語法
WITH ( <table_hint> [ [ , ] ...n ] )
<table_hint> ::=
{ NOEXPAND
| INDEX ( <index_value> [ , ...n ] ) | INDEX = ( <index_value> )
| FORCESEEK [ ( <index_value> ( <index_column_name> [ , ... ] ) ) ]
| FORCESCAN
| HOLDLOCK
| NOLOCK
| NOWAIT
| PAGLOCK
| READCOMMITTED
| READCOMMITTEDLOCK
| READPAST
| READUNCOMMITTED
| REPEATABLEREAD
| ROWLOCK
| SERIALIZABLE
| SNAPSHOT
| SPATIAL_WINDOW_MAX_CELLS = <integer_value>
| TABLOCK
| TABLOCKX
| UPDLOCK
| XLOCK
}
<table_hint_limited> ::=
{
KEEPIDENTITY
| KEEPDEFAULTS
| HOLDLOCK
| IGNORE_CONSTRAINTS
| IGNORE_TRIGGERS
| NOLOCK
| NOWAIT
| PAGLOCK
| READCOMMITTED
| READCOMMITTEDLOCK
| READPAST
| REPEATABLEREAD
| ROWLOCK
| SERIALIZABLE
| SNAPSHOT
| TABLOCK
| TABLOCKX
| UPDLOCK
| XLOCK
}
引數
WITH ( <table_hint> ) [ [ , ] ...n ]
在某些情況下,只有在使用 FROM
關鍵詞指定提示時,WITH
子句才支持數據表提示。 資料表提示也必須用括號來指定。
重要
省略 WITH
關鍵詞是已被取代的功能:此功能將在未來的 SQL Server 版本中移除。 請避免在新的開發工作中使用這項功能,並規劃修改目前使用這項功能的應用程式。
不論是否有 WITH
關鍵字,都允許使用下列資料表提示:NOLOCK
、READUNCOMMITTED
、UPDLOCK
、REPEATABLEREAD
、SERIALIZABLE
、READCOMMITTED
、TABLOCK
、TABLOCKX
、PAGLOCK
、ROWLOCK
、NOWAIT
、READPAST
、XLOCK
、SNAPSHOT
和 NOEXPAND
。 在沒有 WITH
關鍵字的情況下指定這些資料表提示時,應該單獨指定這些提示。 例如:
FROM t (TABLOCK)
使用另一個選項指定提示時,必須使用 WITH
關鍵詞來指定提示:
FROM t WITH (TABLOCK, INDEX(myindex))
我們建議您在資料表提示之間使用逗號。
重要
以空格分隔提示,而不是逗號是已被取代的功能:此功能將在未來的 SQL Server 版本中移除。 請避免在新的開發工作中使用這項功能,並規劃修改目前使用這項功能的應用程式。
NOEXPAND
指定當查詢最佳化工具處理查詢時,不展開任何索引檢視表來存取基礎資料表。 查詢最佳化工具在處理檢視表時,會將它視為具有叢集索引的資料表。
NOEXPAND
只適用於索引檢視表。 如需詳細資訊,請參閱 使用 NOEXPAND。
INDEX ( <index_value> [ , ...n ] ) | INDEX = ( <index_value> )
INDEX()
語法會指定要由查詢最佳化工具在其處理陳述式時使用之一或多個索引的名稱或識別碼。 替代的 INDEX =
語法會指定單一索引值。 每份資料表只能指定一個索引提示。
如果叢集索引存在,INDEX(0)
會強制執行叢集索引掃描,而 INDEX(1)
會強制執行叢集索引掃描或搜尋。 如果沒有叢集索引存在,INDEX(0)
會強制執行資料表掃描,並將 INDEX(1)
解譯為錯誤。
如果在單一提示清單中使用多個索引,便會忽略重複項目,且會利用其餘列出的索引來擷取資料表的資料列。 索引提示中的索引順序非常重要。 另外,多個索引提示也會強制執行索引的 AND 作業,而且查詢最佳化工具會在所存取的每個索引上盡量套用多一點的條件。 如果提示索引的集合不含此查詢所參考的所有資料行,則當 SQL Server 資料庫引擎擷取所有索引資料行之後,將會執行擷取作業來擷取其餘的資料行。
注意
當星狀聯結的事實資料表使用參考多個索引的索引提示時,最佳化工具會忽略索引提示,且會傳回一則警告訊息。 此外,指定索引提示的數據表不允許索引 ORing。
資料表提示中的最大索引數目是 250 個非叢集索引。
KEEPIDENTITY
只有在
指定識別欄位要使用匯入之資料檔案中的一個或多個識別值。 如果未指定 KEEPIDENTITY
,則會驗證此數據行的識別值,但不會匯入,而且查詢優化器會根據數據表建立期間指定的種子和遞增值自動指派唯一值。
重要
如果數據檔不包含資料表或檢視中標識列的值,而且標識列不是數據表中的最後一個數據行,您必須略過標識列。 如需詳細資訊,請參閱 使用格式檔案略過數據欄位 (SQL Server)。 如果順利跳過識別欄位,查詢最佳化工具會自動在匯入的資料表資料列中,指派識別欄位的唯一值。
如需在 INSERT ... SELECT * FROM OPENROWSET(BULK...)
語句中使用這個提示的範例,請參閱 大容量導入資料時保留識別值 (SQL Server)。
如需檢查資料表之識別值的相關信息,請參閱 DBCC CHECKIDENT。
KEEPDEFAULTS
只有在
指定插入數據表數據行的預設值,如果有的話,而不是在數據記錄缺少數據行的值時 NULL
。
如需在 INSERT ... SELECT * FROM OPENROWSET(BULK...)
語句中使用這個提示的範例,請參閱 大容量導入期間保留 null 或預設值。
FORCESEEK [ ( <index_value> ( <index_column_name> [ , ...n ] ) ) ]
指定查詢最佳化工具只使用索引搜尋作業做為資料表或檢視表資料的存取路徑。
注意
從 SQL Server 2008 R2 (10.50.x) Service Pack 1 開始,您也可以指定索引參數。 如此一來,查詢最佳化工具只會考慮至少使用指定索引資料行之指定索引的索引搜尋作業。
index_value
索引名稱或索引識別碼值。 您無法指定索引識別碼 0 (堆積)。 若要傳回索引名稱或識別碼,請查詢
sys.indexes
目錄檢視。index_column_name
要加入搜尋作業的索引資料行名稱。 搭配索引參數指定
FORCESEEK
,類似於搭配FORCESEEK
提示使用INDEX
。 但您可以同時指定要搜尋的索引,以及搜尋作業所要考慮的索引資料行,以更有效地控制查詢最佳化工具所使用的存取路徑。 優化工具可能會視需要考慮更多數據行。 例如,如果指定了非叢集索引,除了指定的數據行之外,優化器可以選擇使用叢集索引鍵數據行。
FORCESEEK
提示可使用下列方式進行指定。
語法 | 範例 | 描述 |
---|---|---|
不搭配索引或 INDEX 提示 |
FROM dbo.MyTable WITH (FORCESEEK) |
查詢最佳化工具只會考慮透過任何相關索引的索引搜尋作業存取資料表或檢視表。 |
結合 INDEX 提示 |
FROM dbo.MyTable WITH (FORCESEEK, INDEX (MyIndex)) |
查詢最佳化工具只會考慮利用指定索引的索引搜尋作業存取資料表或檢視表。 |
透過指定索引和索引資料行參數化 | FROM dbo.MyTable WITH (FORCESEEK (MyIndex (col1, col2, col3))) |
如此一來,查詢最佳化工具只會考慮至少使用指定索引資料行之指定索引的索引搜尋作業存取資料表或檢視。 |
使用 FORCESEEK
提示 (搭配或不搭配索引參數) 時,請考慮下列指導方針:
- 您可以將提示指定為資料表提示或查詢提示。 如需查詢提示的詳細資訊,請參閱 查詢提示 (Transact-SQL)。
- 若要將
FORCESEEK
套用至索引檢視表,也必須指定NOEXPAND
提示。 - 每個資料表或檢視表最多可套用提示一次。
- 您無法指定遠端資料來源的提示。 搭配索引提示指定
FORCESEEK
會傳回錯誤 7377;而不搭配索引提示使用FORCESEEK
則會傳回錯誤 8180。 - 如果
FORCESEEK
找不到計劃,則會傳回錯誤 8622。
搭配索引參數指定 FORCESEEK
時,請注意下列指導方針和限制:
- 無法針對
INSERT
、UPDATE
或DELETE
語句的目標數據表指定提示。 - 此提示無法與
INDEX
提示或其他FORCESEEK
提示同時指定。 - 至少必須指定一個資料行,且該資料行必須是前端索引鍵資料行。
- 您可以指定其他索引資料行,但無法略過索引鍵資料行。 例如,指定的索引如有包含索引鍵資料行
a
、b
和c
,有效的語法是FORCESEEK (MyIndex (a))
和FORCESEEK (MyIndex (a, b)
。 無效的語法則是FORCESEEK (MyIndex (c))
和FORCESEEK (MyIndex (a, c)
。 - 在提示中指定的資料行名稱順序,必須符合參考索引中的資料行順序。
- 您無法指定不在索引鍵定義中的資料行。 例如,在非叢集索引中,您只可指定經過定義的索引鍵資料行。 無法指定自動包含在索引中的叢集索引鍵數據行,但可由優化器使用。
- 您無法指定 xVelocity 記憶體最佳化的資料行存放區索引做為索引參數。 系統會傳回錯誤 366。
- 修改索引定義(例如,加入或移除數據行)可能需要修改參考該索引的查詢。
- 此提示會讓最佳化工具無法使用資料表的任何空間或 XML 索引。
- 此提示無法與
FORCESCAN
提示同時指定。 - 如果是資料分割索引,即無法在
FORCESEEK
提示中指定 SQL Server 隱含加入的資料分割資料行。
警告
搭配參數指定 FORCESEEK
會限制最佳化工具所能使用的計畫數目,其限制幅度將大於不搭配參數指定 FORCESEEK
時的限制。 這可能會導致在更多情況下發生 Plan cannot be generated
錯誤。
FORCESCAN
適用於:SQL Server 2008 R2 (10.50.x) Service Pack 1 和更新版本
指定查詢最佳化工具只使用索引掃描作業做為參考資料表或檢視表的存取路徑。 當最佳化工具低估受影響的資料列數,並選擇搜尋作業而非掃描作業時,使用 FORCESCAN
提示可能有助於查詢。 發生此情況時,授與作業的記憶體數量會太少,而導致查詢效能受到影響。
您可以搭配或不搭配 FORCESCAN
提示指定 INDEX
。 當與索引提示結合時(INDEX = index_name, FORCESCAN
),查詢優化器會在存取參考的數據表時,只考慮透過指定的索引掃描存取路徑。 您可以搭配索引提示 FORCESCAN
指定 INDEX(0)
,對基底資料表強制執行資料表掃描作業。
針對數據分割數據表和索引,在透過查詢述詞評估排除分割區之後,就會套用 FORCESCAN
。 這表示掃描只會套用至其餘分割區,而不會套用到整份資料表。
FORCESCAN
提示有下列限制:
- 無法針對
INSERT
、UPDATE
或DELETE
語句的目標數據表指定提示。 - 此提示無法搭配多個索引提示使用。
- 此提示會讓查詢最佳化工具無法將資料表上的任何空間或 XML 索引納入考量。
- 您無法指定遠端資料來源的提示。
- 此提示無法與
FORCESEEK
提示同時指定。
HOLDLOCK
相當於 SERIALIZABLE
。 如需詳細資訊,請參閱本文稍後的 SERIALIZABLE。
HOLDLOCK
僅適用於其指定之數據表或檢視表,而且僅適用於其所使用的語句所定義的交易期間。
HOLDLOCK
無法在包含 FOR BROWSE
選項的 SELECT 陳述式中使用。
IGNORE_CONSTRAINTS
只有在
指定大容量導入作業會忽略數據表上的任何條件約束。 根據預設,INSERT
會 唯一條件約束和檢查條件約束 和 主要和外鍵條件約束。 針對大容量導入作業指定 IGNORE_CONSTRAINTS
時,INSERT
必須忽略目標數據表上的這些條件約束。 您無法停用 UNIQUE
、PRIMARY KEY
或 NOT NULL
條件約束。
如果輸入數據包含違反條件約束的數據列,您可能會想要停用 CHECK
和 FOREIGN KEY
條件約束。 藉由停用 CHECK
和 FOREIGN KEY
條件約束,您可以匯入數據,然後使用 Transact-SQL 語句來清除數據。
不過,當忽略 CHECK
和 FOREIGN KEY
條件約束時,數據表上的每個忽略條件約束都會在作業之後 is_not_trusted
或 sys.foreign_keys 目錄檢視中標示為 。 您應該在某個點上,檢查整份資料表的條件約束。 如果大容量導入作業之前數據表不是空的,重新驗證條件約束的成本可能會超過套用 CHECK
的成本,並將條件約束 FOREIGN KEY
條件約束套用至累加數據的成本。
IGNORE_TRIGGERS
只有在
指定大量匯入作業忽略資料表所定義的任何觸發程序。 根據預設,INSERT
會套用觸發程式。
只有在應用程式不依賴任何觸發程式且將效能最大化時,才使用 IGNORE_TRIGGERS
。
NOLOCK
相當於 READUNCOMMITTED
。 如需詳細資訊,請參閱本文稍後的 READUNCOMMITTED。
注意
針對 UPDATE
或 DELETE
語句:此功能將在未來的 SQL Server 版本中移除。 請避免在新的開發工作中使用這項功能,並規劃修改目前使用這項功能的應用程式。
NOWAIT
指示資料庫引擎一旦發現資料表的鎖定,便立即傳回訊息。
NOWAIT
相當於針對特定資料表指定 SET LOCK_TIMEOUT 0
。 同時包含 NOWAIT
提示時,TABLOCK
提示無效。 如果在使用 TABLOCK
提示時要終止查詢而不等候,請改為在查詢前面加上 SET LOCK_TIMEOUT 0;
。
PAGLOCK
在通常會採用資料列或索引鍵的個別鎖定,或通常會採用單一資料表鎖定的情況下,頁面會鎖定。 依預設,會使用作業所適用的鎖定模式。 如果是在以 SNAPSHOT
隔離等級操作的交易中指定時,不會採用頁面鎖定,除非 PAGLOCK
是與其他需要鎖定的資料表提示相結合,例如 UPDLOCK
和 HOLDLOCK
。
READCOMMITTED
指定讀取作業符合 READ COMMITTED
隔離等級的規則,方法是使用鎖定或數據列版本設定。 如果資料庫選項 READ_COMMITTED_SNAPSHOT
OFF
,Database Engine 會取得共用鎖定,因為數據已讀取,並在讀取作業完成時釋放這些鎖定。 如果資料庫選項 READ_COMMITTED_SNAPSHOT
ON
,則 Database Engine 不會取得鎖定並使用數據列版本設定。 如需隔離等級的詳細資訊,請參閱 SET TRANSACTION ISOLATION LEVEL。
注意
針對 UPDATE
或 DELETE
語句:此功能將在未來的 SQL Server 版本中移除。 請避免在新的開發工作中使用這項功能,並規劃修改目前使用這項功能的應用程式。
READCOMMITTEDLOCK
指定讀取作業符合 READ COMMITTED
隔離等級的規則,方法是使用鎖定。 不論資料庫選項 READ_COMMITTED_SNAPSHOT
的設定為何,資料庫引擎一律在讀取資料時取得共用鎖定,並在讀取作業完成時釋放這些鎖定。 如需隔離等級的詳細資訊,請參閱 SET TRANSACTION ISOLATION LEVEL。 無法在 INSERT
語句的目標數據表上指定這個提示;傳回錯誤 4140。
READPAST
指定資料庫引擎不讀取其他交易已鎖定的資料列。 當指定 READPAST
時,系統會略過資料列層級鎖定,但不會略過頁面層級鎖定。 換句話說,資料庫引擎會略過資料列,而不會封鎖目前的交易,直到釋放鎖定為止。 例如,假設資料表 T1
包含了值為 1、2、3、4、5 的單一整數資料行。 如果交易 A 將 3 的值變更為 8,但尚未認可, SELECT * FROM T1 (READPAST)
則會產生值 1、2、4、5。
READPAST
主要是在實作使用 SQL Server 資料表的工作佇列時用來減少鎖定競爭。 使用 READPAST
的佇列讀取器會略過其他交易已鎖定的佇列項目,直接到下一個可用的佇列項目,不需要等待其他交易釋放鎖定。
READPAST
可以指定 UPDATE
或 DELETE
語句中參考的任何數據表,以及 FROM
子句中參考的任何數據表。 在 UPDATE
語句中指定時,只有在讀取數據以識別要更新的記錄時,才套用 READPAST
,不論在語句中指定的位置為何。
READPAST
無法在 INTO
語句的 INSERT
子句中指定數據表。 更新或刪除使用 READPAST
的作業,可能會在讀取外鍵或索引檢視表,或修改次要索引時封鎖。
READPAST
只能在作業於 READ COMMITTED
或 REPEATABLE READ
隔離等級的交易中指定。 如果是在以 SNAPSHOT
隔離等級操作的交易中指定時,READPAST
必須與其他需要鎖定的資料表提示相結合,例如 UPDLOCK
和 HOLDLOCK
。
當 [READPAST
資料庫] 選項設定為 [READ_COMMITTED_SNAPSHOT
],且下列任一條件為 true 時,就無法指定 ON
數據表提示:
- 工作階段的交易隔離等級
READ COMMITTED
。 - 查詢中也有指定
READCOMMITTED
資料表提示。
若要在這些情況下指定 READPAST
提示,請移除 READCOMMITTED
資料表提示 (如果有的話),並在查詢中包含 READCOMMITTEDLOCK
資料表提示。
READUNCOMMITTED
指定允許中途讀取。 不會發出任何共用鎖定來防止其他交易修改目前交易所讀取的資料,其他交易所設定的獨佔鎖定也不會封鎖目前交易,使它無法讀取鎖定的資料。 允許中途讀取可以提高並行程度,但代價是所讀取的資料修改後來會被其他交易回復。 這可能會為您的交易產生錯誤、向使用者呈現從未認可過的數據,或導致使用者看到記錄兩次(或完全不見)。
READUNCOMMITTED
和 NOLOCK
提示只適用於資料鎖定。 所有查詢,包括具有 READUNCOMMITTED
和 NOLOCK
提示的查詢,在編譯和執行期間取得 Sch-S(架構穩定性)鎖定。 因此,當並行交易在資料表上保有 Sch-M (結構描述修改) 鎖定時,查詢將會遭到封鎖。 例如,資料定義語言 (DDL) 作業會在修改資料表的結構描述資訊之前先取得 Sch-M 鎖定。 嘗試取得 Sch-S 鎖定時,會封鎖任何並行查詢,包括使用 READUNCOMMITTED
或 NOLOCK
提示執行的查詢。 相反地,保有 Sch-S 鎖定的查詢將會封鎖嘗試取得 Sch-M 鎖定的並行交易。
無法對插入、更新或刪除作業修改的資料表指定 READUNCOMMITTED
和 NOLOCK
。 SQL Server 查詢優化器會忽略套用至 READUNCOMMITTED
或 NOLOCK
語句目標數據表之 FROM
子句中的 UPDATE
和 DELETE
提示。
注意
在未來的 SQL Server 版本中,將會移除適用於 READUNCOMMITTED
或 NOLOCK
語句目標數據表的 FROM
子句中 UPDATE
和 DELETE
提示的支援。 請避免在新的開發工作中使用此內容中的這些提示,並規劃修改目前在使用這些提示的應用程式。
您可以使用下列任一選項,將鎖定爭用降至最低,同時保護交易免於讀取未認可的數據修改:
-
READ COMMITTED
設定READ_COMMITTED_SNAPSHOT
資料庫選項的ON
隔離等級。 -
SNAPSHOT
隔離等級。
如需隔離等級的詳細資訊,請參閱 SET TRANSACTION ISOLATION LEVEL。
注意
如果您在指定 時收到 READUNCOMMITTED
,請解決死結錯誤 (錯誤訊息 1205),然後重試您的語句。
REPEATABLEREAD
指定掃描是以與 REPEATABLE READ
隔離等級執行之交易相同的鎖定語意來執行。 如需隔離等級的詳細資訊,請參閱 SET TRANSACTION ISOLATION LEVEL。
ROWLOCK
指定通常在採用頁面或資料表鎖定時,採用資料列鎖定。 如果是在以 SNAPSHOT
隔離等級操作的交易中指定時,不會採用資料列鎖定,除非 ROWLOCK
是與其他需要鎖定的資料表提示相結合,例如 UPDLOCK
和 HOLDLOCK
。
ROWLOCK
無法搭配具有叢集資料行存放區索引的資料表使用。 下列範例會將錯誤 651 傳回應用程式。
UPDATE [dbo].[FactResellerSalesXL_CCI] WITH (ROWLOCK)
SET UnitPrice = 50
WHERE ProductKey = 150;
SERIALIZABLE
相當於 HOLDLOCK
。 使共用鎖定更具限制性的方法是將共用鎖定持續保留到交易完成為止,而不是在不再需要所要求的資料表或資料頁面時,便立即釋放共用鎖定 (不論交易是否完成)。 利用與在 SERIALIZABLE
隔離等級執行之交易相同的語意來執行掃描。 如需隔離等級的詳細資訊,請參閱 SET TRANSACTION ISOLATION LEVEL。
SNAPSHOT
適用於:SQL Server 2014 (12.x) 和更新版本
記憶體最佳化的資料表是在 SNAPSHOT
隔離下存取。
SNAPSHOT
只能搭配記憶體最佳化的資料表使用 (無法搭配磁碟式資料表使用),如下列範例所示。 如需詳細資訊,請參閱記憶體最佳化的資料表簡介。
SELECT *
FROM dbo.Customers AS c WITH (SNAPSHOT)
LEFT OUTER JOIN dbo.[Order History] AS oh
ON c.customer_id = oh.customer_id;
SPATIAL_WINDOW_MAX_CELLS = <integer_value>
適用於:SQL Server 2012 (11.x) 和更新版本
指定要用於鑲嵌幾何或地理物件的資料格數上限。 <integer_value> 是介於 1 和 8192 之間的值。
此選項允許藉由調整主要和次要篩選執行時間之間的取捨,微調查詢執行時間。 較大的數字會降低次要篩選執行時間,但增加主要篩選執行時間,較小的數字會減少主要篩選執行時間,但增加次要篩選器執行時間。 對於較密集的空間資料,較高的數字應該會藉由使用主要篩選提供更好的近似值,並減少次要篩選執行時間,產生更快速的執行時間。 如需更多疏鬆數據,較低的數位會減少主要篩選運行時間。
這個選項適用於手動和自動方格鑲嵌。
TABLOCK
指定在資料表層級套用取得的鎖定。 取得的鎖定類型取決於執行的陳述式。 例如,SELECT
語句可能會取得共享鎖定。 指定 TABLOCK
可以將共用鎖定套用至整份資料表,而不會在資料列或頁面層級進行套用。 如果同時指定了 HOLDLOCK
,就會將資料表鎖定保留到交易結束為止。
使用 INSERT INTO <target_table> SELECT <columns> FROM <source_table>
語句將數據匯入堆積時,您可以藉由指定 TABLOCK
目標數據表的提示,為語句啟用最低限度記錄和最佳鎖定。 此外,資料庫的復原模式必須設定為簡單或大量記錄。
TABLOCK
提示也可啟用平行插入至堆積或叢集資料行存放區索引。 如需詳細資訊,請參閱 INSERT。
當搭配 OPENROWSET 大量資料列集提供者使用,以將資料匯入資料表時,TABLOCK
會使多個用戶端能夠以最佳化的記錄和鎖定,將資料並行載入目標資料表中。 如需詳細資訊,請參閱 大容量導入中最低限度記錄的必要條件。
TABLOCKX
指定獨佔鎖定是在資料表上取得。
UPDLOCK
指定採用更新鎖定,且保留到交易完成為止。
UPDLOCK
只會在資料列層級或頁面層級,為讀取作業採用更新鎖定。 如果 UPDLOCK
與 TABLOCK
結合,或基於其他原因而採用數據表層級鎖定,則會改為採用獨佔式 (X) 鎖定。
如有指定 UPDLOCK
,將會忽略 READCOMMITTED
和 READCOMMITTEDLOCK
隔離等級提示。 例如,如果工作階段的隔離等級設定為 SERIALIZABLE
,而查詢指定了 (UPDLOCK
, READCOMMITTED
),將會忽略 READCOMMITTED
提示,並使用 SERIALIZABLE
隔離等級執行交易。
XLOCK
指定採用獨佔鎖定,且保留到交易完成為止。 如果指定了 ROWLOCK
、PAGLOCK
或 TABLOCK
,就會將獨佔鎖定套用在適當的資料粒度層級上。
備註
如果查詢計劃並未存取資料表,就會忽略資料表提示。 這可能是優化器選擇完全不存取數據表,或是因為改為存取索引檢視表所造成。 在後者的情況下,可以使用 OPTION (EXPAND VIEWS)
查詢提示來防止存取索引檢視表。
所有鎖定提示都會傳播到查詢計劃所存取的所有資料表和檢視表,包括檢視表中所參考的資料表和檢視表。 另外,SQL Server 也會執行對應的鎖定一致性檢查。
取得數據列層級鎖定的鎖定提示 ROWLOCK
、UPDLOCK
和 XLOCK
,可能會鎖定索引鍵,而不是實際的數據列。 例如,如果數據表具有非叢集索引,而且使用鎖定提示的 SELECT
語句是由涵蓋索引來處理,則會在涵蓋索引的索引鍵上取得鎖定,而不是在基表的數據列上取得鎖定。
如果資料表包含計算資料行,且計算資料行是由存取其他資料表資料行的運算式或函數來計算,資料表提示就不會在這些資料表上使用,也不會傳播。 例如,在查詢中指定資料表的 NOLOCK
資料表提示。 這份資料表擁有多個計算資料行,這些計算資料行會利用存取另一資料表中之資料行的運算式和函數的組合來進行計算。 當存取運算式和函數所參考的資料表時,這些資料表不會使用 NOLOCK
資料表提示。
SQL Server 不允許下列每個資料表針對 FROM
子句中的每個數據表,提供一個以上的數據表提示:
- 資料粒度提示:
PAGLOCK
、NOLOCK
、READCOMMITTEDLOCK
、ROWLOCK
、TABLOCK
或TABLOCKX
。 - 隔離等級提示:
HOLDLOCK
、NOLOCK
、READCOMMITTED
、REPEATABLEREAD
、SERIALIZABLE
。
已篩選的索引提示
篩選的索引可以當做數據表提示使用,但如果查詢未涵蓋查詢選取的所有數據列,則會導致查詢優化器產生錯誤 8622。 下列是無效之已篩選的索引提示範例。 此範例會建立篩選的索引 FIBillOfMaterialsWithComponentID
,然後使用它作為 SELECT
語句的索引提示。 已篩選的索引述詞包括 ComponentID 533、324 和 753 的資料列。 此查詢述詞也包含 ComponentID 533、324 和 753 的資料列,但會擴充結果集,使其包含 ComponentID 855 和 924 (這兩者不在已篩選的索引中)。 因此,查詢最佳化工具無法使用已篩選的索引提示,而且會產生錯誤 8622。 如需詳細資訊,請參閱 建立篩選索引。
IF EXISTS (SELECT name FROM sys.indexes
WHERE name = N'FIBillOfMaterialsWithComponentID'
AND object_id = OBJECT_ID(N'Production.BillOfMaterials'))
DROP INDEX FIBillOfMaterialsWithComponentID
ON Production.BillOfMaterials;
GO
CREATE NONCLUSTERED INDEX [FIBillOfMaterialsWithComponentID]
ON Production.BillOfMaterials(ComponentID, StartDate, EndDate)
WHERE ComponentID IN (533, 324, 753);
GO
SELECT StartDate, ComponentID
FROM Production.BillOfMaterials
WITH (INDEX (FIBillOfMaterialsWithComponentID))
WHERE ComponentID IN (533, 324, 753, 855, 924);
GO
如果 SET
選項沒有篩選索引的必要值,查詢優化器就不會考慮索引提示。 如需詳細資訊,請參閱 CREATE INDEX。
使用 NOEXPAND
NOEXPAND
只適用於「索引檢視表」。 索引檢視表是建立了唯一叢集索引的檢視表。 如果查詢包含同時在索引檢視表和基底資料表中的資料行參考,而且查詢最佳化工具判斷使用索引檢視表能夠提供最好的查詢執行方法,則查詢最佳化工具會使用檢視表的索引。 此功能稱為索引檢視表比對。 查詢優化器自動使用索引檢視表只有在 SQL Server 的特定版本中才支援。 Azure SQL 資料庫和 Azure SQL 受控執行個體支援在不指定 NOEXPAND
提示的情況下自動使用索引檢視表。
如需詳細資訊,請參閱查詢處理架構指南。
如需 SQL Server 版本所支援的功能清單,請參閱:
- SQL Server 2022 的版本和支援功能
- SQL Server 2019 的版本及支援功能
- 版本及支援的 SQL Server 2017 功能
- 版本及支援的 SQL Server 2016 功能
不過,若要讓查詢優化器考慮索引檢視進行比對,或使用以 NOEXPAND
提示參考的索引檢視表,則必須將下列 SET
選項設定為 ON
。
- ANSI_NULLS
- ANSI_PADDING
- ANSI_WARNINGS
- ARITHABORT 1
- CONCAT_NULL_YIELDS_NULL
- QUOTED_IDENTIFIER
當 設定為 ARITHABORT
時,ON
ANSI_WARNINGS
會隱含地設定為 ON
。 因此,您不需要手動調整這個設定。
此外,NUMERIC_ROUNDABORT
選項必須設定為 OFF
。
若要強制查詢最佳化工具使用索引檢視表的索引,請指定 NOEXPAND
選項。 僅當查詢中指定了檢視的名稱時,才可使用此提示。 SQL Server 不會提供提示來強制特定索引檢視用於未直接在 FROM
子句中命名檢視的查詢中。 不過,即使查詢中未直接參考索引檢視表,查詢最佳化工具仍會考慮使用索引檢視表。 使用 NOEXPAND
數據表提示時,SQL Server Database Engine 只會在索引檢視表上自動建立統計數據。 省略此提示可能會導致遺漏統計資料的相關執行計畫警告,該警告無法透過手動建立統計資料來解決。
在查詢優化期間,資料庫引擎 會在查詢直接參考檢視並使用提示時,使用自動或手動建立的NOEXPAND
檢視統計數據。
將資料表提示當做查詢提示使用
數據表提示 也可以使用 OPTION (TABLE HINT)
子句來指定為查詢提示。 我們建議您只在 計劃指南的內容中,才將資料表提示當做查詢提示使用。 如果是特定的查詢,只將這些提示指定為資料表提示。 如需詳細資訊,請參閱
權限
KEEPIDENTITY
、IGNORE_CONSTRAINTS
和 IGNORE_TRIGGERS
提示需要數據表的 ALTER
許可權。
範例
A. 使用 TABLOCK 提示指定鎖定方法
下列範例會指定在 AdventureWorks2022 資料庫中的 Production.Product
數據表上取得共享鎖定,並保留到 UPDATE
語句結尾為止。
UPDATE Production.Product WITH (TABLOCK)
SET ListPrice = ListPrice * 1.10
WHERE ProductNumber LIKE 'BK-%';
GO
B. 使用 FORCESEEK 提示指定索引搜尋作業
下列範例會 FORCESEEK
使用提示而不指定索引,強制查詢優化器在 AdventureWorks2022 資料庫中的數據表上 Sales.SalesOrderDetail
執行索引搜尋作業。
SELECT *
FROM Sales.SalesOrderHeader AS h
INNER JOIN Sales.SalesOrderDetail AS d WITH (FORCESEEK)
ON h.SalesOrderID = d.SalesOrderID
WHERE h.TotalDue > 100
AND (d.OrderQty > 5
OR d.LineTotal < 1000.00);
GO
下列範例會搭配索引使用 FORCESEEK
提示,強制查詢最佳化工具對指定的索引和索引資料行執行索引搜尋作業。
SELECT h.SalesOrderID,
h.TotalDue,
d.OrderQty
FROM Sales.SalesOrderHeader AS h
INNER JOIN Sales.SalesOrderDetail AS d
WITH (FORCESEEK (PK_SalesOrderDetail_SalesOrderID_SalesOrderDetailID (SalesOrderID)))
ON h.SalesOrderID = d.SalesOrderID
WHERE h.TotalDue > 100
AND (d.OrderQty > 5
OR d.LineTotal < 1000.00);
GO
C. 使用 FORCESCAN 提示指定索引掃描作業
下列範例會 FORCESCAN
使用提示來強制查詢優化器在 AdventureWorks2022 資料庫中的數據表上 Sales.SalesOrderDetail
執行掃描作業。
SELECT h.SalesOrderID,
h.TotalDue,
d.OrderQty
FROM Sales.SalesOrderHeader AS h
INNER JOIN Sales.SalesOrderDetail AS d WITH (FORCESCAN)
ON h.SalesOrderID = d.SalesOrderID
WHERE h.TotalDue > 100
AND (d.OrderQty > 5
OR d.LineTotal < 1000.00);