Note
Access to this page requires authorization. You can try signing in or changing directories.
Access to this page requires authorization. You can try changing directories.
Applies to:
SQL Server 2025 (17.x) Preview
Azure SQL Database
Azure SQL Managed Instance
SQL database in Microsoft Fabric Preview
Optimized locking offers an improved transaction locking mechanism to reduce lock blocking and lock memory consumption for concurrent transactions.
What is optimized locking?
Optimized locking helps to reduce lock memory as very few locks are held even for large transactions. In addition, optimized locking avoids lock escalations and can avoid certain types of deadlocks. This allows more concurrent access to the table.
Optimized locking is composed of two primary components: transaction ID (TID) locking and lock after qualification (LAQ).
- A transaction ID (TID) is a unique identifier of a transaction. Each row is labeled with the last TID that modified it. Instead of potentially many key or row identifier locks, a single lock on the TID is used. For more information, see Transaction ID (TID) locking.
- Lock after qualification (LAQ) is an optimization that evaluates query predicates using the latest committed version of the row without acquiring a lock, thus improving concurrency. LAQ requires read committed snapshot isolation (RCSI). For more information, see Lock after qualification (LAQ).
For example:
- Without optimized locking, updating 1,000 rows in a table might require 1,000 exclusive (
X
) row locks held until the end of the transaction. - With optimized locking, updating 1,000 rows in a table might require 1,000
X
row locks but each lock is released as soon as each row is updated, and only one TID lock is held until the end of the transaction. Because locks are released quickly, lock memory usage is reduced and lock escalation is much less likely to occur, improving workload concurrency.
Note
Enabling optimized locking reduces or eliminates row and page locks acquired by the Data Modification Language (DML) statements such as INSERT
, UPDATE
, DELETE
, MERGE
. It has no effect on other kinds of database and object locks, such as schema locks.
Availability
The following table summarizes the availability and enabled state of optimized locking across SQL platforms.
Platform | Available | Enabled by default |
---|---|---|
Azure SQL Database | Yes | Yes (always enabled) |
SQL database in Microsoft Fabric Preview | Yes | Yes (always enabled) |
Azure SQL Managed InstanceAUTD | Yes | Yes (always enabled) |
Azure SQL Managed Instance2022 | No | N/A |
SQL Server 2025 (17.x) Preview | Yes | No (can be enabled per database) |
SQL Server 2022 (16.x) and older versions | No | N/A |
Enable and disable
To enable or disable optimized locking for a SQL Server database, use the ALTER DATABASE ... SET OPTIMIZED_LOCKING = ON | OFF
command. For more information, see ALTER DATABASE SET options.
Optimized locking builds on other database features:
- You must enable accelerated database recovery (ADR) on a database before you can enable optimized locking. Conversely, to disable ADR, you must disable optimized locking first if it's enabled.
- For the most benefit from optimized locking, read committed snapshot isolation (RCSI) should be enabled for the database. The LAQ component of optimized locking is in effect only if RCSI is enabled.
In Azure SQL Database and Azure SQL Managed Instance, ADR is always enabled and RCSI is enabled by default.
To verify that these options are enabled for your current database, connect to the database and run the following T-SQL query:
SELECT database_id,
name,
is_accelerated_database_recovery_on,
is_read_committed_snapshot_on,
is_optimized_locking_on
FROM sys.databases
WHERE name = DB_NAME();
Is optimized locking enabled?
Optimized locking is enabled per user database. Connect to your database, then use the following query to check if optimized locking is enabled:
SELECT DATABASEPROPERTYEX(DB_NAME(), 'IsOptimizedLockingOn') AS is_optimized_locking_enabled;
Result | Description |
---|---|
0 |
Optimized locking is disabled. |
1 |
Optimized locking is enabled. |
NULL |
Optimized locking isn't available. |
You can also use the sys.databases catalog view. For example, to see if optimized locking is enabled for all databases, execute the following query:
SELECT database_id,
name,
is_optimized_locking_on
FROM sys.databases;
Locking overview
This is a short summary of the behavior when optimized locking isn't enabled. For more information, review the Transaction locking and row versioning guide.
In the database engine, locking is a mechanism that prevents multiple transactions from updating the same data simultaneously in order to guarantee the ACID properties of transactions.
When a transaction needs to modify data, it requests a lock on the data. The lock is granted if no other conflicting locks are held on the data, and the transaction can proceed with the modification. If another conflicting lock is held on the data, the transaction must wait for the lock to be released before it can proceed.
When multiple transactions attempt to access the same data concurrently, the database engine must resolve potentially complex conflicts with concurrent reads and writes. Locking is one of the mechanisms by which the engine can provide the semantics for the ANSI SQL transaction isolation levels. Although locking in databases is essential, reduced concurrency, deadlocks, complexity, and lock overhead can affect performance and scalability.
Transaction ID (TID) locking
When row versioning based isolation levels are in use or when ADR is enabled, every row in the database internally contains a transaction ID (TID). This TID is persisted on disk. Every transaction modifying a row stamps that row with its TID.
With TID locking, instead of taking the lock on the key of the row, a lock is taken on the TID of the row. The modifying transaction holds an X
lock on its TID. Other transactions acquire an S
lock on the TID to wait until the first transaction completes. With TID locking, page and row locks continue to be taken for modifications, but each page and row lock is released as soon as each row is modified. The only lock held until the end of transaction is the single X
lock on the TID resource, replacing multiple page and row (key) locks.
Consider the following example that shows locks for the current session while a write transaction is active:
/* Is optimized locking is enabled? */
SELECT DATABASEPROPERTYEX(DB_NAME(), 'IsOptimizedLockingOn') AS is_optimized_locking_enabled;
CREATE TABLE t0
(
a int PRIMARY KEY,
b int NULL
);
INSERT INTO t0 VALUES (1,10),(2,20),(3,30);
GO
BEGIN TRANSACTION;
UPDATE t0
SET b = b + 10;
SELECT *
FROM sys.dm_tran_locks
WHERE request_session_id = @@SPID
AND
resource_type IN ('PAGE','RID','KEY','XACT');
COMMIT TRANSACTION;
GO
DROP TABLE IF EXISTS t0;
If optimized locking is enabled, the request holds only a single X
lock on the XACT
(transaction) resource.
If optimized locking isn't enabled, the same request holds four locks - one IX
(intent exclusive) lock on the page containing the rows, and three X
key locks on each row:
The sys.dm_tran_locks dynamic management view (DMV) is useful in examining or troubleshooting locking issues, such as observing optimized locking in action.
Lock after qualification (LAQ)
Building on the TID infrastructure, the LAQ component of optimized locking changes how DML statements such as INSERT
, UPDATE
, and DELETE
acquire locks.
Without optimized locking, query predicates are checked row by row in a scan by first taking an update (U
) row lock. If the predicate is satisfied, an exclusive (X
) row lock is taken before updating the row and held until the end of transaction.
With optimized locking, and when the READ COMMITTED
snapshot isolation level (RCSI) is enabled, predicates can be optimistically checked on latest committed version of the row without taking any locks. If the predicate doesn't satisfy, the query moves to the next row in the scan. If the predicate is satisfied, an X
row lock is taken to update the row.
In other words, the lock is taken after qualification of the row for modification. The X
row lock is released as soon as the row update is complete, before the end of the transaction.
Since predicate evaluation is performed without acquiring any locks, concurrent queries modifying different rows don't block each other.
For example:
/* Confirm that optimized locking and read committed snapshot isolation (RCSI) are both enabled on this database. */
SELECT database_id,
name,
is_accelerated_database_recovery_on,
is_optimized_locking_on,
is_read_committed_snapshot_on
FROM sys.databases
WHERE name = DB_NAME();
CREATE TABLE t1
(
a int NOT NULL,
b int NULL
);
INSERT INTO t1
VALUES (1,10),(2,20),(3,30);
GO
Session 1 | Session 2 |
---|---|
BEGIN TRANSACTION; UPDATE t1 SET b = b + 10 WHERE a = 1; |
|
BEGIN TRANSACTION; UPDATE t1 SET b = b + 10 WHERE a = 2; |
|
COMMIT TRANSACTION; |
|
COMMIT TRANSACTION; |
Without optimized locking, session 2 is blocked because session 1 holds a U
lock on the row session 2 needs to update. However, with optimized locking, session 2 isn't blocked because U
locks aren't taken, and because in the latest committed version of row 1, column a
equals to 1, which doesn't satisfy the predicate of session 2.
LAQ is performed optimistically on the assumption that a row isn't modified after checking the predicate. If the predicate is satisfied and the row hasn't been modified after checking the predicate, it's modified by the current transaction.
Because U
locks aren't taken, a concurrent transaction might modify the row after the predicate has been evaluated. If there's an active transaction holding an X
TID lock on the row, the database engine waits for it to complete. If the row has changed after the predicate was evaluated previously, the database engine re-evaluates (re-qualifies) the predicate again before modifying the row. If the predicate is still satisfied, the row is modified.
Predicate re-qualification is supported by a subset of the query engine operators. If predicate re-evaluation is needed, but the query plan uses an operator that doesn't support predicate re-qualification, the database engine internally aborts statement processing and restarts it without LAQ. When such an abort occurs, the lock_after_qual_stmt_abort
extended event fires.
Some statements, for example UPDATE
statements with variable assignment and statements with the OUTPUT clause, can't be aborted and restarted without changing their semantics. For such statements, LAQ isn't used.
In the following example, the predicate is re-evaluated because another transaction has changed the row:
CREATE TABLE t3
(
a int NOT NULL,
b int NULL
);
INSERT INTO t3 VALUES (1,10),(2,20),(3,30);
GO
Session 1 | Session 2 |
---|---|
BEGIN TRANSACTION; UPDATE t3 SET b = b + 10 WHERE a = 1; |
|
BEGIN TRANSACTION; UPDATE t3 SET b = b + 10 WHERE a = 1; |
|
COMMIT TRANSACTION; |
|
COMMIT TRANSACTION; |
LAQ heuristics
As described in Lock after qualification (LAQ), when LAQ is used, some statements might be internally restarted and processed without LAQ. If this happens frequently, the overhead of repeated processing might become significant. To keep this overhead to a minimum, optimized locking uses a heuristics mechanism to track repeated processing. This mechanism disables LAQ for the database if the overhead exceeds a threshold.
For the purposes of the heuristics mechanism, the work done by a statement is measured in the number of pages it has processed (logical reads). If the database engine is modifying a row that has been modified by another transaction after statement processing started, then the work done by the statement is treated as potentially wasted because the statement might be aborted and restarted. The system keeps track of the total potentially wasted work and the total work done by all statements in the database.
LAQ is disabled for the database if the percentage of the potentially wasted work exceeds a threshold. LAQ is also disabled if the number of restarted statements exceeds a threshold.
If the wasted work and the number of restarted statements fall below their respective thresholds, LAQ is re-enabled for the database.
LAQ limitations
Lock after qualification might not be used in the following scenarios:
- When disabled by LAQ heuristics.
- When conflicting locking hints, such as
UPDLOCK
,READCOMMITTEDLOCK
,XLOCK
, orHOLDLOCK
are used. - When the transaction isolation level is other than
READ COMMITTED
, or when theREAD_COMMITTED_SNAPSHOT
database option is disabled. - When the table being modified has a columnstore index.
- When the DML statement includes variable assignment.
- When the DML statement has an
OUTPUT
clause. - When the DML statement uses more than one index seek or scan operator to read the rows being modified.
- In
MERGE
statements.
Query behavior changes with optimized locking and RCSI
Concurrent workloads under read committed snapshot isolation (RCSI) that rely on strict execution order of transactions might experience differences in query behavior when optimized locking is enabled.
Consider the following example where transaction T2 is updating table t4
based on column b
that was updated during transaction T1.
CREATE TABLE t4
(
a int NOT NULL,
b int NULL
);
INSERT INTO t4
VALUES (1,1);
GO
Session 1 | Session 2 |
---|---|
BEGIN TRANSACTION T1; UPDATE t4 SET b = 2 WHERE a = 1; |
|
BEGIN TRANSACTION T2; UPDATE t4 SET b = 3 WHERE b = 2; |
|
COMMIT TRANSACTION; |
|
COMMIT TRANSACTION; |
Let's evaluate the outcome of the previous scenario with and without lock after qualification (LAQ).
Without LAQ
Without LAQ, the UPDATE
statement in transaction T2 is blocked, waiting for transaction T1 to complete. Once T1 completes, T2 updates the row setting column b
to 3
because its predicate is satisfied.
After both transactions commit, table t4
contains the following rows:
a | b
1 | 3
With LAQ
With LAQ, transaction T2 uses the latest committed version of the row where column b
equals to 1
to evaluate its predicate (b = 2
). The row doesn't qualify; hence it's skipped and the statement completes without having been blocked by transaction T1. In this example, LAQ removes blocking but leads to different results.
After both transactions commit, table t4
contains the following rows:
a | b
1 | 2
Important
Even without LAQ, applications shouldn't assume that the database engine guarantees strict ordering without using locking hints when row versioning based isolation levels are used. Our general recommendation for customers running concurrent workloads under RCSI that rely on strict execution order of transactions (as shown in the previous example) is to use stricter isolation levels such as REPEATABLE READ
and SERIALIZABLE
.
Diagnostic additions for optimized locking
The following improvements help you monitor and troubleshoot blocking and deadlocks when optimized locking is enabled:
- Wait types for optimized locking
XACT
wait types for theS
lock on the TID, and resource descriptions in sys.dm_os_wait_stats:LCK_M_S_XACT_READ
- Occurs when a task is waiting for a shared lock on anXACT
wait_resource
type, with an intent to read.LCK_M_S_XACT_MODIFY
- Occurs when a task is waiting for a shared lock on anXACT
wait_resource
type, with an intent to modify.LCK_M_S_XACT
- Occurs when a task is waiting for a shared lock on anXACT
wait_resource
type, where the intent can't be inferred. This scenario isn't common.
- Locking resources visibility
XACT
locking resources. For more information, seeresource_description
in sys.dm_tran_locks.
- Wait resource visibility
XACT
wait resources. For more information, seewait_resource
in sys.dm_exec_requests.
- Deadlock graph
- Under each resource in the deadlock report
<resource-list>
, each<xactlock>
element reports the underlying resources and specific information for locks of each member of a deadlock. For more information and an example, see Optimized locking and deadlocks.
- Under each resource in the deadlock report
- Extended events
- The
lock_after_qual_stmt_abort
event fires when a statement is internally aborted and restarted because of a conflict with another transaction. For more information, see Lock after qualification (LAQ). - In SQL Server 2025 (17.x) Preview and Azure SQL Managed Instance, the
locking_stats
event fires for every database every several minutes and provides aggregate locking statistics for the time interval, such as the number of lock escalations, whether TID locking and LAQ components of optimized locking are enabled, and the number of queries where LAQ wasn't used for various reasons. This event fires even if optimized locking is disabled.
- The
Best practices with optimized locking
Enable read committed snapshot isolation (RCSI)
To maximize the benefits of optimized locking, it's recommended to enable read committed snapshot isolation (RCSI) on the database and use READ COMMITTED
isolation as the default isolation level. If not already enabled, enable RCSI by connecting to the master
database and executing the following statement:
ALTER DATABASE [database-name-placeholder] SET READ_COMMITTED_SNAPSHOT ON;
In Azure SQL Database, RCSI is enabled by default and READ COMMITTED
is the default isolation level. With RCSI enabled and when using READ COMMITTED
isolation level, readers read a version of the row from the snapshot taken at the start of the statement. With LAQ, writers qualify rows per the predicate based on the latest committed version of the row and without acquiring U
locks. With LAQ, a query waits only if the row qualifies and there's an active write transaction on that row. Qualifying based on the latest committed version and locking only the qualified rows reduces blocking and increases concurrency.
In addition to reduced blocking, the required lock memory is reduced. This is because readers don't take any locks, and writers take only short duration locks, instead of locks that are held until the end of the transaction. When you use stricter isolation levels such as REPEATABLE READ
or SERIALIZABLE
, the database engine holds row and page locks until the end of the transaction even with optimized locking enabled, for both readers and writers, resulting in increased blocking and lock memory usage.
Avoid locking hints
While table and query hints such as UPDLOCK
, READCOMMITTEDLOCK
, XLOCK
, HOLDLOCK
, etc. are honored when optimized locking is enabled, they reduce the benefit of optimized locking. Lock hints force the database engine to take row or page locks and hold them until the end of the transaction, to honor the intent of the lock hints. Some applications have logic where lock hints are needed, for example when reading a row with the UPDLOCK
hint and then updating it later. We recommend using lock hints only where needed.
With optimized locking, there are no restrictions on existing queries and queries don't need to be rewritten. Queries that aren't using hints benefit from optimized locking most.
A table hint on one table in a query doesn't disable optimized locking for other tables in the same query. Further, optimized locking only affects the locking behavior of tables being updated by a DML statement such as INSERT
, UPDATE
, DELETE
, or MERGE
. For example:
CREATE TABLE t5
(
a int NOT NULL,
b int NOT NULL
);
CREATE TABLE t6
(
a int NOT NULL,
b int NOT NULL
);
GO
INSERT INTO t5 VALUES (1,10),(2,20),(3,30);
INSERT INTO t6 VALUES (1,10),(2,20),(3,30);
GO
UPDATE t5 SET t5.b = t6.b
FROM t5
INNER JOIN t6 WITH (UPDLOCK)
ON t5.a = t6.a;
In the previous query example, only table t6
is affected by the locking hint, while t5
can still benefit from optimized locking.
UPDATE t5
SET t5.b = t6.b
FROM t5 WITH (REPEATABLEREAD)
INNER JOIN t6
ON t5.a = t6.a;
In the previous query example, only table t5
uses the REPEATABLE READ
isolation level and hold locks until the end of the transaction. Other updates to t5
can still benefit from optimized locking. The same applies to the HOLDLOCK
hint.
Frequently asked questions (FAQ)
Is optimized locking on by default in both new and existing databases?
In Azure SQL Database, Azure SQL Managed InstanceAUTD, and SQL database in Microsoft Fabric Preview, yes. In SQL Server 2025 (17.x) Preview optimized locking is disabled by default but can be enabled on any user database that has accelerated database recovery enabled.
How can I detect if optimized locking is enabled?
See Is optimized locking enabled?
What if I want to force queries to block despite optimized locking?
If RCSI is enabled, use the READCOMMITTEDLOCK
table hint to force blocking between two queries when optimized locking is enabled.
Is optimized locking used on read-only secondary replicas?
No, because DML statements can't run on read-only replicas, and the corresponding row and page locks aren't taken.
Is optimized locking used when modifying data in tempdb and in temporary tables?
Not at this time.