with(nolock)的用法

本文导读:要提升SQL的查询效能,一般来说大家会以建立索引(index)为第一考虑。其实除了index的建立之外,当我们在下SQL Command时,在语法中加一段WITH (NOLOCK)可以改善在线大量查询的环境中数据集被LOCK的现象藉此改善查询的效能。不过有一点千万要注意的就是,WITH (NOLOCK)的SQL SELECT有可能会造成Dirty Read。

with(nolock)的介绍

大家在写查询时,为了性能,往往会在表后面加一个nolock,或者是with(nolock),其目的就是查询是不锁定表,从而达到提高查询速度的目的。

当同一时间有多个用户访问同一资源,并发用户中如果有用户对资源做了修改,此时就会对其它用户产生某些不利的影响,例如:

1、脏读

一个用户对一个资源做了修改,此时另外一个用户正好读取了这条被修改的记录,然后,第一个用户放弃修改,数据回到修改之前,这两个不同的结果就是脏读。

2、不可重复读

一个用户的一个操作是一个事务,这个事务分两次读取同一条记录,如果第一次读取后,有另外用户修改了这个数据,然后第二次读取的数据正好是其它用户修改的数据,这样造成两次读取的记录不同,如果事务中锁定这条记录就可以避免。

3、幻读

指用户读取一批记录的情况,用户两次查询同一条件的一批记录,第一次查询后,有其它用户对这批数据做了修改,方法可能是修改,删除,新增,第二次查询时,会发现第一次查询的记录条目有的不在第二次查询结果中,或者是第二次查询的条目不在第一次查询的内容中。

NOLOCK 语句执行时不发出共享锁,允许脏读 ,等于 READ UNCOMMITTED事务隔离级别 。nolock确实在查询时能提高速度,但它并不是没有缺点的,起码它会引起脏读、只适用与select查询语句。 在一些不需要考虑脏读的场合会用到,例如当用户在论坛发广告贴时删除其所有发帖,这个查询就不怕脏读,全删,或者漏一个正在发的都不是问题。

二、实例

SQL 代码 复制

SELECT COUNT(UserID)

FROM EMPLOYEE WITH (NOLOCK)

JOIN WORKING_GROUP WITH (NOLOCK)

ON EMPLOYEE.UserID = WORKING_GROUP.UserID

三、with(nolock)的使用场景

1:数据量特别大的表,牺牲数据安全性来提升性能是可以考虑的;

2:允许出现脏读现象的业务逻辑,反之一些数据完整性要求比较严格的场景就不合适了,像金融方面等。

3:数据不经常修改的表,这样会省于锁定表的时间来大大加快查询速度。

4、当使用NoLock时,它允许阅读那些已经修改但是还没有交易完成的数据。因此如果有需要考虑transaction事务数据的实时完整性时,使用WITH (NOLOCK)就要好好考虑一下。

四、nolock和with(nolock)的几个小区别

1、SQL05中的同义词,只支持with(nolock);

2、with(nolock)的写法非常容易再指定索引。

3、跨服务器查询语句时 不能用with (nolock) 只能用nolock,同一个服务器查询时 则with(nolock)和nolock都可以用
————————————————
版权声明:本文为CSDN博主「朱智文」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/u014236541/article/details/80001673

可以使用 MyBatis 的拦截器来实现在查询 SQL 语句中添加 WITH(NOLOCK) 的功能。以下是一个简单的实现步骤: 1. 定义一个实现了 `Interceptor` 接口的类,例如 `TableWithNoLockInterceptor`。 2. 在 `TableWithNoLockInterceptor` 中实现 `intercept` 方法,该方法会在执行 SQL 语句时被调用。在该方法中,可以获取到当前执行的 SQL 语句,然后添加 WITH(NOLOCK) 后再执行 SQL 语句。 3. 使用 `@Intercepts` 和 `@Signature` 注解来将 `TableWithNoLockInterceptor` 注册到 MyBatis 的拦截器链中。 以下是一个简单的实现示例: ```java @Intercepts({ @Signature(type = StatementHandler.class, method = "prepare", args = { Connection.class, Integer.class }) }) public class TableWithNoLockInterceptor implements Interceptor { private static final String WITH_NOLOCK = " WITH(NOLOCK)"; private static final String SELECT = "SELECT "; private static final Pattern TABLE_PATTERN = Pattern.compile("(FROM|JOIN)\\s+(\\[?\\S+\\]?)+\\s+"); @Override public Object intercept(Invocation invocation) throws Throwable { StatementHandler statementHandler = (StatementHandler) invocation.getTarget(); MetaObject metaObject = SystemMetaObject.forObject(statementHandler); MappedStatement mappedStatement = (MappedStatement) metaObject.getValue("delegate.mappedStatement"); String sql = (String) metaObject.getValue("delegate.boundSql.sql"); if (mappedStatement.getSqlCommandType() == SqlCommandType.SELECT) { Matcher matcher = TABLE_PATTERN.matcher(sql); StringBuffer sb = new StringBuffer(); while (matcher.find()) { String table = matcher.group(2); if (!table.contains(WITH_NOLOCK)) { matcher.appendReplacement(sb, matcher.group().replace(table, table + WITH_NOLOCK)); } } matcher.appendTail(sb); sql = sb.toString(); metaObject.setValue("delegate.boundSql.sql", sql); } return invocation.proceed(); } } ``` 在以上示例中,我们实现了一个拦截器 `TableWithNoLockInterceptor`,它会在执行查询 SQL 语句时自动添加 WITH(NOLOCK)。在 `intercept` 方法中,我们首先获取到当前执行的 SQL 语句,并判断该语句是否为查询语句,如果是,则使用正则表达式找到所有的表名,并在表名后添加 WITH(NOLOCK),最后执行 SQL 语句。 需要注意的是,以上示例仅供参考,实际使用时可能需要根据具体情况进行修改和优化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值