SQLSERVER 事务日志的 LSN 到底是什么?
一:背景
1. 讲故事
大家都知道数据库应用程序 它天生需要围绕着数据文件打转,诸如包含数据的 .mdf,事务日志的 .ldf,很多时候深入了解这两类文件的合成原理,差不多对数据库就能理解一半了,关于 .mdf 的合成前面的文章已经有所介绍,这篇我们来聊一下 .ldf 的一些内部知识,比如 LSN。
二:对 LSN 的理解
1. 什么是 LSN
如果大家玩过 SQLSERVER 的发布订阅或者 AlwaysOn 或多或少都见过 LSN,比如下面的格式: 00000030:00018090:0002 ,这一串编号到底是什么意思呢?本质上指示的是 .ldf 文件的某一个物理位置上的偏移,画个图大概如下:
从图中可以看到其实是由 虚拟文件号:日志段起始扇区编号:槽号编号 三部分组成,要了解这三部分就需要明白 .ldf 文件是如何进行逻辑划分的,画个简图如下:
通过上面的图很容易就能明白其中的逻辑关系,事务日志文件被划分成了多个 虚拟文件,虚拟文件又划分成了多个 日志段,日志段又划分成了多个 扇区,日志段中日志记录位置存储在 槽号 中,有了这些理论基础,接下来用一个案例来加深大家的理解吧。
2. 一个案例演示
新建一个 MyLSN 数据库,再创建一个 test 表,插入 3w 条记录,sql如下:
CREATE DATABASE MyLSN
GO
USE MyLSN
GO
CREATETABLE test(a INTIDENTITY, b CHAR(10) DEFAULT'aaaaaaaaaa')SET NOCOUNT ONINSERTINTO test (b) DEFAULTVALUES
GO 30000SET NOCOUNT OFF
接下来通过 fn_dblog 来查询和 dbo.test 表相关的事务日志记录。
SELECT [Current LSN],Operation,Context,AllocUnitName,[RowLog Contents 0],[Log Record],[Log Record Length]
FROM fn_dblog(NULL, NULL)
WHERE AllocUnitName LIKE'%test%';
从图中可以看到这是一个 INSERT 的事务日志记录,这里就拿编号 00000030:00000db0:0002 去定位 .ldf 中的物理偏移位置吧,要想获取物理偏移就要知道下面偏移值才可以。
0x30 虚拟文件号的偏移值是多少 ?
要想知道这个信息,可以用 DBCC loginfo 命令,查看 FSeqNo下的 StartOffset 偏移值即可,即 0n48 对应的 4071424 ,截图如下:
0xdb0 扇区号的偏移是多少?
大家都知道磁盘的扇区是 512byte,sqlserver 为了更好的写入磁盘,也用了 512byte 这个粒度,所以偏移值就是 512 * 0xdb0。
综合上面就能定位到日志段的物理偏移值为:
lkd> ?0n4071424 + (0n3504*0n512)
Evaluate expression: 5865472 = 00000000`00598000
接下来用 WinHex 来定位 MyLSN_log.ldf 文件偏移 00598000 的位置,定位之前先将数据库离线。
ALTER DATABASE MyLSN SET OFFLINE
前面的 0x0003 表示该日志段只有 3 条记录,后面的 0x019E 表示该日志段的大小为 414byte,接下来就是槽号了,槽号位置的物理偏移计算规则如下:
lkd> ? 00598000 + 019E - 1
Evaluate expression: 5865885 = 00000000`0059819d
从图中可以看到,slot2 的偏移值为 00C8,即物理偏移值为 005980c8 。
lkd> ? 00598000 + 00C8
Evaluate expression: 5865672 = 00000000`005980c8
从上面框出的内容可以轻松的看到,事务日志中记录了 Insert 的 aaaaaaaaaa 值,太棒了,起始就是 fn_dblog 查出来的 Log Record 值。
三:总结
对 LSN 有一个深度的理解,对各种数据库事务日志暴涨的故障分析都会有一个很好的理论基础,后面我们再聊这些话题。
SQLSERVER 事务日志的 LSN 到底是什么?相关推荐
- 解释一下SQLSERVER事务日志记录
解释一下SQLSERVER事务日志记录 大家知道在完整恢复模式下,SQLSERVER会记录每个事务所做的操作,这些记录会存储在事务日志里,有些软件会利用事务日志来读取 操作记录恢复数据,例如:log ...
- (转)解释一下SQLSERVER事务日志记录
本文转载自桦仔的博客http://www.cnblogs.com/lyhabc/archive/2013/07/16/3194220.html 解释一下SQLSERVER事务日志记录 大家知道在完整恢 ...
- sqlserver 事务日志已满和'PRIMARY'
sqlserver 事务日志已满和'PRIMARY' 1.出现这个问题是日志文件达到了单个文件的极限了要解决这个问题有两个方法 1.删除日志:解决方法是这样子的:DUMP TRANSACTION 库名 ...
- sqlserver 事务日志过大 收缩方法解决方案
sqlserver 事务日志过大 收缩方法解决方案 参考文章: (1)sqlserver 事务日志过大 收缩方法解决方案 (2)https://www.cnblogs.com/QingKing/p/4 ...
- SQLSERVER 事务日志 LSN 到底是什么?
一:背景 1. 讲故事 大家都知道数据库应用程序 它天生需要围绕着数据文件打转,诸如包含数据的 .mdf,事务日志的 .ldf,很多时候深入了解这两类文件的合成原理,差不多对数据库就能理解一半了,关于 ...
- 收缩sqlserver事务日志
若要允许 DBCC SHRINKFILE 命令收缩文件,首先需要通过将数据库恢复模式设置为 SIMPLE 来截断该文件. 示例,收缩数据库abce的事务日志 USE abce; GO -- Trunc ...
- SQLserver 事务日志已满解决方法
问题: 数据库 'MSDB' 的事务日志已满.若要查明无法重用日志中的空间的原因,请参阅 sys.databases 中的 log_reuse_wait_desc 列. 解决方法: 在tempdb库属 ...
- sqlserver 事务日志 异常增长原因排查_小白入门学习打日志
前言 只有光头才能变强. 文本已收录至我的GitHub仓库,欢迎Star:https://github.com/ZhongFuCheng3y/3y 记得之前写过一篇:<阿里巴巴 Java开发手册 ...
- SQLSERVER事务日志已满 the transaction log for database 'xx' is full
解决办法:清除日志 USE [master] GO ALTER DATABASE DNName SET RECOVERY SIMPLE WITH NO_WAIT GOALTER DATABASE DN ...
最新文章
- Java常见内存溢出(OOM)解决方案
- Python中函数的参数传递方式
- 《计算机算法设计与分析》题目汇总
- 跟我学ModelArts丨探索ModelArts平台个性化联邦学习API
- Fiddler——模拟限速
- python有什么内容_python的类(简介,没什么内容)
- windows xp下Apache2.2.11整合Tomcat6.0.20(集群模式无集群模式)
- chroot环境的快速构建
- Hadoop集群部署模型纵览3
- 如何写一份优秀的java程序员简历
- 2019年第十二届中国大学生计算机设计大赛总结
- [娱乐]帝国时代I:罗马复兴秘籍
- Java-----关于IO流的总结
- 80c51单片机编程语言是什么,80C51单片机的指令系统(一)程序设计语言和指令...
- GB/T 10707 橡胶燃烧性能
- 架构师之spring------@Autowire注入多泛型实例 can not cast to的问题解决
- 【Simapro软件教程】参数进行灵敏度分析
- 【JavaScript】内容的展开/收起
- HTC 重力感应传感器编程资料总结
- 【MongoDB】——数据存储结构与基本数据类型