1
1
using Microsoft . EntityFrameworkCore . Diagnostics ;
2
2
using System . Data . Common ;
3
3
using System . Text . RegularExpressions ;
4
+ using Microsoft . EntityFrameworkCore ;
4
5
5
6
namespace Masuit . Tools . Core ;
6
7
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
8
13
{
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 ) ;
10
15
11
16
public override InterceptionResult < object > ScalarExecuting ( DbCommand command , CommandEventData eventData , InterceptionResult < object > result )
12
17
{
13
- command . CommandText = TableAliasRegex . Replace (
14
- command . CommandText ,
15
- "${tableAlias} WITH (NOLOCK)"
16
- ) ;
18
+ AddWithNoLock ( command ) ;
17
19
return base . ScalarExecuting ( command , eventData , result ) ;
18
20
}
19
21
@@ -22,31 +24,24 @@ public override InterceptionResult<object> ScalarExecuting(DbCommand command, Co
22
24
public override Task < InterceptionResult < object > > ScalarExecutingAsync ( DbCommand command , CommandEventData eventData , InterceptionResult < object > result ,
23
25
CancellationToken cancellationToken = new CancellationToken ( ) )
24
26
{
25
- command . CommandText = TableAliasRegex . Replace (
26
- command . CommandText ,
27
- "${tableAlias} WITH (NOLOCK)"
28
- ) ;
27
+ AddWithNoLock ( command ) ;
29
28
return base . ScalarExecutingAsync ( command , eventData , result , cancellationToken ) ;
30
29
}
31
30
32
31
#else
32
+
33
33
public override ValueTask < InterceptionResult < object > > ScalarExecutingAsync ( DbCommand command , CommandEventData eventData , InterceptionResult < object > result ,
34
34
CancellationToken cancellationToken = new CancellationToken ( ) )
35
35
{
36
- command . CommandText = TableAliasRegex . Replace (
37
- command . CommandText ,
38
- "${tableAlias} WITH (NOLOCK)"
39
- ) ;
36
+ AddWithNoLock ( command ) ;
40
37
return base . ScalarExecutingAsync ( command , eventData , result , cancellationToken ) ;
41
38
}
39
+
42
40
#endif
43
41
44
42
public override InterceptionResult < DbDataReader > ReaderExecuting ( DbCommand command , CommandEventData eventData , InterceptionResult < DbDataReader > result )
45
43
{
46
- command . CommandText = TableAliasRegex . Replace (
47
- command . CommandText ,
48
- "${tableAlias} WITH (NOLOCK)"
49
- ) ;
44
+ AddWithNoLock ( command ) ;
50
45
return result ;
51
46
}
52
47
@@ -55,22 +50,38 @@ public override InterceptionResult<DbDataReader> ReaderExecuting(DbCommand comma
55
50
public override Task < InterceptionResult < DbDataReader > > ReaderExecutingAsync ( DbCommand command , CommandEventData eventData , InterceptionResult < DbDataReader > result ,
56
51
CancellationToken cancellationToken = new CancellationToken ( ) )
57
52
{
58
- command . CommandText = TableAliasRegex . Replace (
59
- command . CommandText ,
60
- "${tableAlias} WITH (NOLOCK)"
61
- ) ;
53
+ AddWithNoLock ( command ) ;
62
54
return base . ReaderExecutingAsync ( command , eventData , result , cancellationToken ) ;
63
55
}
64
56
65
57
#else
58
+
66
59
public override ValueTask < InterceptionResult < DbDataReader > > ReaderExecutingAsync ( DbCommand command , CommandEventData eventData , InterceptionResult < DbDataReader > result ,
67
60
CancellationToken cancellationToken = new CancellationToken ( ) )
68
61
{
69
- command . CommandText = TableAliasRegex . Replace (
70
- command . CommandText ,
71
- "${tableAlias} WITH (NOLOCK)"
72
- ) ;
62
+ AddWithNoLock ( command ) ;
73
63
return base . ReaderExecutingAsync ( command , eventData , result , cancellationToken ) ;
74
64
}
75
65
#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
+ }
76
75
}
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
+ { }
0 commit comments