Skip to content

Commit a5d9044

Browse files
committed
优化WithNoLockInterceptor
1 parent 0f60211 commit a5d9044

File tree

3 files changed

+45
-30
lines changed

3 files changed

+45
-30
lines changed

Directory.Build.props

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
<Project>
22
<PropertyGroup>
3-
<Version>2025.1.5</Version>
3+
<Version>2025.1.6</Version>
44
<Deterministic>true</Deterministic>
55
</PropertyGroup>
66
</Project>
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,21 @@
11
using Microsoft.EntityFrameworkCore.Diagnostics;
22
using System.Data.Common;
33
using System.Text.RegularExpressions;
4+
using Microsoft.EntityFrameworkCore;
45

56
namespace Masuit.Tools.Core;
67

7-
public class QueryWithNoLockDbCommandInterceptor : DbCommandInterceptor
8+
/// <summary>
9+
/// WITH (NOLOCK)全局拦截器,仅限SQL Server使用
10+
/// </summary>
11+
/// <param name="enableGlobalNolock">全局启用,无需手动调用WithNolock扩展</param>
12+
public class WithNoLockInterceptor(bool enableGlobalNolock = false) : DbCommandInterceptor
813
{
9-
private static readonly Regex TableAliasRegex = new Regex(@"(?<tableAlias>AS \[[a-zA-Z]\w*\](?! WITH \(NOLOCK\)))", RegexOptions.Multiline | RegexOptions.Compiled | RegexOptions.IgnoreCase);
14+
private static readonly Regex TableRegex = new Regex(@"(?<table>\[\w+\] AS \[[a-zA-Z]\w*\](?! WITH \(NOLOCK\)))", RegexOptions.Multiline | RegexOptions.Compiled | RegexOptions.IgnoreCase);
1015

1116
public override InterceptionResult<object> ScalarExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<object> result)
1217
{
13-
command.CommandText = TableAliasRegex.Replace(
14-
command.CommandText,
15-
"${tableAlias} WITH (NOLOCK)"
16-
);
18+
AddWithNoLock(command);
1719
return base.ScalarExecuting(command, eventData, result);
1820
}
1921

@@ -22,31 +24,24 @@ public override InterceptionResult<object> ScalarExecuting(DbCommand command, Co
2224
public override Task<InterceptionResult<object>> ScalarExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<object> result,
2325
CancellationToken cancellationToken = new CancellationToken())
2426
{
25-
command.CommandText = TableAliasRegex.Replace(
26-
command.CommandText,
27-
"${tableAlias} WITH (NOLOCK)"
28-
);
27+
AddWithNoLock(command);
2928
return base.ScalarExecutingAsync(command, eventData, result, cancellationToken);
3029
}
3130

3231
#else
32+
3333
public override ValueTask<InterceptionResult<object>> ScalarExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<object> result,
3434
CancellationToken cancellationToken = new CancellationToken())
3535
{
36-
command.CommandText = TableAliasRegex.Replace(
37-
command.CommandText,
38-
"${tableAlias} WITH (NOLOCK)"
39-
);
36+
AddWithNoLock(command);
4037
return base.ScalarExecutingAsync(command, eventData, result, cancellationToken);
4138
}
39+
4240
#endif
4341

4442
public override InterceptionResult<DbDataReader> ReaderExecuting(DbCommand command, CommandEventData eventData, InterceptionResult<DbDataReader> result)
4543
{
46-
command.CommandText = TableAliasRegex.Replace(
47-
command.CommandText,
48-
"${tableAlias} WITH (NOLOCK)"
49-
);
44+
AddWithNoLock(command);
5045
return result;
5146
}
5247

@@ -55,22 +50,38 @@ public override InterceptionResult<DbDataReader> ReaderExecuting(DbCommand comma
5550
public override Task<InterceptionResult<DbDataReader>> ReaderExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<DbDataReader> result,
5651
CancellationToken cancellationToken = new CancellationToken())
5752
{
58-
command.CommandText = TableAliasRegex.Replace(
59-
command.CommandText,
60-
"${tableAlias} WITH (NOLOCK)"
61-
);
53+
AddWithNoLock(command);
6254
return base.ReaderExecutingAsync(command, eventData, result, cancellationToken);
6355
}
6456

6557
#else
58+
6659
public override ValueTask<InterceptionResult<DbDataReader>> ReaderExecutingAsync(DbCommand command, CommandEventData eventData, InterceptionResult<DbDataReader> result,
6760
CancellationToken cancellationToken = new CancellationToken())
6861
{
69-
command.CommandText = TableAliasRegex.Replace(
70-
command.CommandText,
71-
"${tableAlias} WITH (NOLOCK)"
72-
);
62+
AddWithNoLock(command);
7363
return base.ReaderExecutingAsync(command, eventData, result, cancellationToken);
7464
}
7565
#endif
66+
67+
private void AddWithNoLock(DbCommand command)
68+
{
69+
// 检查查询是否有标记
70+
if (enableGlobalNolock || command.CommandText.StartsWith("-- NOLOCK"))
71+
{
72+
command.CommandText = TableRegex.Replace(command.CommandText, "${table} WITH (NOLOCK)");
73+
}
74+
}
7675
}
76+
77+
public static class WithNoLockExt
78+
{
79+
public static IQueryable<T> WithNolock<T>(this IQueryable<T> queryable)
80+
{
81+
return queryable.TagWith("-- NOLOCK");
82+
}
83+
}
84+
85+
[Obsolete("请使用WithNoLockInterceptor替代")]
86+
public class QueryWithNoLockDbCommandInterceptor : WithNoLockInterceptor
87+
{ }

README.md

+7-3
Original file line numberDiff line numberDiff line change
@@ -1181,15 +1181,19 @@ var allchanges=dbContext.GetAllChanges();//获取增删改的实体字段信息
11811181

11821182
#### nolock查询
11831183

1184-
sqlserver
1184+
SQL Server
11851185

11861186
上下文注入Interceptor即可在任何查询时使用nolock查询
11871187

11881188
```csharp
1189-
services.AddDbContext<TContext>(opt => opt.UseSqlserver("ConnString", builder => builder.AddInterceptors(new QueryWithNoLockDbCommandInterceptor()));
1189+
services.AddDbContext<TContext>(opt => opt.UseSqlserver("ConnString", builder => builder.AddInterceptors(new WithNoLockInterceptor(true))); // 启用全局nolock查询
1190+
services.AddDbContext<TContext>(opt => opt.UseSqlserver("ConnString", builder => builder.AddInterceptors(new WithNoLockInterceptor())); // 按需启用全局nolock查询
1191+
1192+
// 按需启用全局nolock查询,执行单个nolock查询
1193+
await dbContext.Users.Where(x=>x.Name=="aaa").WithNolock().ToListAsync();
11901194
```
11911195

1192-
通用数据库
1196+
其他通用数据库
11931197

11941198
nolock本质是开启一个 `读未提交`级别的事务,此时的查询性能最好,但有可能会读取到脏数据。
11951199

0 commit comments

Comments
 (0)