SQL优化中的重要概念:锁定

 

上篇文章讲的是事务,这篇就引出另一个重要概念,就是锁定。

当一个用户要读取另一个用户正在修改的数据,或者一个用户正在修改另一个用户正在读取的数据,或者一个用户要修改另一个用户正在修改的数据,就会出现并发问题。锁定能防止并发问题。

资源的锁定方式称为锁定模式,SQL Server中的锁定模式:共享锁,意向锁,更新锁,排他锁,架构稳定锁,架构修改锁,大批量更新锁,键范围锁。不是所有锁模式都是兼容的,如:一个加了排他锁的资源不能再加其他锁,其他事务必须等待,直到释放排他锁。

可以锁定SQL Server中的各类对象,可以锁定的资源在粒度上差异很大,从细粒度(行、键)到粗粒度(数据库)。细粒度的锁允许用户能查询那些未被锁定的行,并发性更高,但是需要更多的锁资源(每个被锁定的行都需要一个锁资源);粗粒度的锁降低了并发性,但需要的锁资源很少。

 

1、在SQL Server中可锁定的资源:

DB(数据库)
	Metadata(系统元数据)
	Object(数据库对象:视图,函数,存储过程,触发器)
	Table(表)
		Hobt(堆或B树)
			Allocation Unit(按照数据的类型(数据,行溢出、大对象)分组的相关页面)
				Extent(8个8KB的页面)
					Page(8KB数据页面)
						Rid(行标示符对应一个堆表的行)
						Key(键范围上的锁、B树中的键)
File
Application

 

2、查看锁的活动

select resource_type,                  --资源类型
       resource_database_id,           --资源所在的数据库id
       resource_associated_entity_id,  --数据库中与资源相关联的实体的 ID。
                                       --该值可以是对象ID、Hobt ID 或分配单元 ID,
                                       --具体视资源类型而定
       object_name(resource_associated_entity_id,resource_database_id),
       
       resource_lock_partition,  --已分区锁资源的锁分区ID。对于未分区锁资源值为 0       
       resource_description,  --资源的说明,其中只包含从其他资源列中无法获取的信息
       
       request_session_id,    --请求资源的会话
       request_type,          --请求类型,该值为 LOCK       
       request_mode,      --请求的模式,对于已授予的请求,为已授予模式,
                          --对于等待请求,为正在请求的模式(锁定模式)                              
       request_status         --请求的当前状态,
                              --可能值为 GRANTED、CONVERT 或 WAIT
       
from sys.dm_tran_locks
WHERE request_session_id = 361

 

3、控制表的锁升级

每个锁都会消耗内存资源,当锁的数量增加时,那么所需要的内存就会增加,而系统内可用的内存就会减少。如果锁占用的内存比率超过一个阀值,SQL Server会将细粒度锁(行锁)升级为粗粒度锁(表锁),这个过程就是锁升级。

 

锁升级的优点是可以减少锁的数量,相应的减少内存的使用量,而缺点是由于锁住了更大的资源,所以会导致阻塞,降低并发性。

--默认值,不管是不是分区表,会在表级别启用锁升级
ALTER TABLE t
SET (lock_escalation = TABLE)

--当表升级时,如果表已经分区,会在分区级别启用锁升级
ALTER TABLE t
SET (lock_escalation = auto)

--在表级别禁用锁升级,如果用了TabLock提示或在Serializable隔离级别下查询,还是会有表锁
ALTER TABLE t
SET (lock_escalation = disable)

4、隔离级别

影响锁定的除了上面提到的锁定模式、锁的粒度,还有就是事务的隔离级别。

所谓隔离级别其实就是事务与事务之间相互影响的程度,比如,一个事务修改了数据,那么其他事务是否能看到这些修改的数据,无论事务是否提交。对于最高的隔离级别,这个事务所做的修改,其他任何事务都看不到;而最低的隔离级别,这个事务所做的修改,可以被其他任何事务看到。

SQL Server隔离级别:

(1)read uncommitted能解决丢失更新的问题,但是会导致脏读。

(2)read committed读取的是已提交的数据,所以解决了脏读的问题,但是会有不可重复读取的问题,也就是在一个事务中有两次读取,第一次读取的和第二次读取的同一条数据,可能值是不同的,因为在事务中的select语句在读取完之后就立即释放的共享锁,而此时有另一个事务把刚才第一个事务读取的那条数据修改了,这样第一次读和第二次读到的值就会不同。

(3)repeatable read解决了不可重复读取的问题,也就是在一个事务中的前后两次读取,读取到的数据值是一样的,但是会有幻读的可能,也就是第一次读出的数据确实和第二次读取的数据一样,但是第二次读取的记录条数可能多于第一次读取的记录条数,因为在读取的时候确实是锁住了被读取的记录,但是这个表可能添加了新的记录。

(4)serializable通过锁住查询范围内的键、键与键之间的范围来解决幻读的问题,比如where id >=5 and id <=10,加入表表中只有id为7,9的两条记录,那么5-6、7-8、9-10这3个范围都会被锁住。


(5)在ALLOW_SNAPSHOT_ISOLATION下的snapshot这种隔离级别允许读取事务一致性版本的数据,但可能不是最新的版本,也就是说在一个事务中只能读到某个版本,比如,在一个事务中有两次读取,第一次读完后,数据被另一个事务修改且事务提交了,此时进行第2次读取,那么读出来的还是和第一次读取一样的数据,这就是在一个事务中如果数据被其他事务修改了,读出来的数据也一样。优点是数据读取不会阻塞写,写也不会阻塞读取。另外,如果两个事务同时修改同一行数据,会导致更新冲突错误。

(6)在READ_COMMITTED_SNAPSHOT下的read committed隔离级别允许在同一事务中总是能读取运行的已提交的数据,而且数据读取不会阻塞写,写也不会阻塞读取,也不会导致更新冲突。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
这个PDF文件是我花钱买来的,现在为了挣积分,拿出来与大家分享!! 本书深入浅出地介绍了目前世界上最受欢迎的数据库管理系统之一——SQL Server。全书共分三个部分:第一部分阐释了数据库的基本概念,讲解了数据库建模语言;第二部分展示了从概念建模到在SQL Server 2008上真正实现数据库的过程;第三部分深入探讨了SQL Server若干方面的技术细节,如数据保护、索引、并发访问等。通过将理论融入数据库实践,清晰地讲解了关系型数据库的设计原则,完整地展示了如何进行良好的关系型数据库设计,深入揭示了SQL Server 2008的技术细节。   本书浓缩了作者作为SQL Server数据库架构师多年来丰富的实践经验,适合各类数据库开发和管理人员学习参考 目录 第1章 数据库概念简介  1.1 数据库设计阶段   1.1.1 概念阶段   1.1.2 逻辑阶段   1.1.3 实现阶段   1.1.4 物理阶段  1.2 关系数据结构   1.2.1 数据库和模式   1.2.2 表、行和列   1.2.3 信息原则   1.2.4 域   1.2.5 元数据   1.2.6 键   1.2.7 未显式赋值的项(NULL)  1.3 实体之间的关系   1.3.1 二元关系   1.3.2 非二元关系  1.4 数据访问语言(SQL)  1.5 理解依赖性   1.5.1 函数依赖性   1.5.2 判定  1.6 总结 第2章 数据建模语言  2.1 数据建模介绍  2.2 实体  2.3 属性   2.3.1 主键   2.3.2 替代键   2.3.3 外键   2.3.4 域   2.3.5 命名  2.4 关系   2.4.1 识别性关系   2.4.2 非识别性关系   2.4.3 角色名字   2.4.4 关系基数   2.4.5 动词短语(关系名字)  2.5 描述信息  2.6 其他建模方法   2.6.1 信息工程   2.6.2 Chen ERD   2.6.3 Visio   2.6.4 Management Studio数据库关系图  2.7 最佳实践  2.8 总结 第3章 概念阶段数据建模  3.1 理解需求  3.2 文档化过程  3.3 需求收集   3.3.1 客户访谈   3.3.2 要回答的问题   3.3.3 现存的系统和原型   3.3.4 其他类型的文档  3.4 识别对象和过程   3.4.1 识别实体   3.4.2 实体间关系   3.4.3 识别属性和域  3.5 识别业务规则和业务过程   3.5.1 识别业务规则   3.5.2 识别基础业务过程  3.6 完成概念模型   3.6.1 识别明显的、额外的数据需求   3.6.2 和客户一起评审   3.6.3 重复以上步骤直到客户同意你的模型  3.7 最佳实践  3.8 总结 第4章 规范化过程  4.1 为什么要规范化   4.1.1 消灭重复数据   4.1.2 避免编写不必要的代码   4.1.3 给表瘦身   4.1.4 最大化聚集索引的使用   4.1.5 降低每张表索引的数量  4.2 规范化应该走多远  4.3 规范化过程  4.4 实体和属性的形式:第一范式   4.4.1 所有属性必须是原子的   4.4.2 实体的所有实例必须包含相同数量的值   4.4.3 实体出现的所有实体类型都必须不同   4.4.4 第一范式所避免的不规则编程   4.4.5 当前设计不符合第一范式的线索  4.5 属性间的关系   4.5.1 第二范式   4.5.2 第三范式   4.5.3 Boyce-Codd范式  4.6 实体的多值依赖   4.6.1 第四范式   4.6.2 第五范式  4.7 非规范化  4.8 最佳实践  4.9 总结  4.10 额外的例子  4.11 本书迄今为止所讲述的故事 第5章 实现基础的表结构  5.1 评审逻辑设计  5.2 变换设计   5.2.1 选择名字   5.2.2 处理子类型   5.2.3 决定树的实现方式   5.2.4 选择键的实现方式   5.2.5 决定域的实现方式   5.2.6 设置模式   5.2.7 评审“最终的”实现模型  5.3 实现设计   5.3.1 创建基本表结构   5.3.2 添加唯一性约束   5.3.3 构建默认约束   5.3.4 添加关系(外键)   5.3.5 处理排序规则和排序   5.3.6 计算列   5.3.7 实现用户定义的数据类型   5.3.8 文档化你的数据库   5.3.9 处理依赖信息  5.4 最佳实践  5.5 总结 第6章 保护数据的完整性  6.1 最佳实
在MySQL进行SQL优化可以提高查询性能和系统的整体效率。以下是一些常见的MySQL SQL优化技巧: 1. 索引优化: - 确保表的列使用合适的索引,特别是在经常用于WHERE和JOIN条件的列上创建索引。 - 避免创建过多的索引,因为每个索引都需要占用存储空间,并且在插入、更新和删除操作时会增加额外的开销。 - 定期进行索引维护,如重新构建索引、优化索引碎片等。 2. 查询优化: - 使用合适的查询语句,尽量避免全表扫描。使用WHERE子句、JOIN和子查询等来限制结果集的大小。 - 避免在查询使用通配符%,因为它会导致全表扫描。如果需要模糊匹配,可以考虑使用全文索引或者其他优化方法。 - 选择合适的字段,只查询需要的列,避免不必要的数据传输和处理。 3. 表结构优化: - 根据业务需求合理设计表结构,避免冗余字段和表的关联过多。 - 合理划分和拆分大表,以减少查询时的数据量。 - 使用合适的数据类型,减小存储空间和提高查询效率。 4. 配置优化: - 调整MySQL的配置参数,如缓冲区大小、连接数、并发数等,以适应实际的负载和需求。 - 合理设置慢查询日志和查询缓存,以便定位慢查询和重复查询。 5. 避免不必要的操作: - 避免在查询使用SELECT *,只选择所需的列。 - 尽量减少数据库的写操作,如更新和删除操作,因为它们会增加额外的开销和锁定需要根据具体情况进行优化,并且通过分析慢查询日志和使用MySQL自带的性能工具来定位问题并进行优化。同时,注意不要过度优化,应根据实际需求和性能要求来平衡优化与开发的成本。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值