热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

当数据库出现页损坏或校验和出错时如何处理

当数据库出现页损坏或校验和出错时如何处理作者:nzperfect/perfectaction日期:2009.09.27Email:nzperfect@gmail.com最近一直在进一步学习数据库故障的处理方面的知识,做为一个数据库维护

  当数据库出现页损坏或校验和出错时如何处理

  作者:nzperfect / perfectaction

  日期:2009.09.27

  Email:nzperfect@gmail.com

  最近一直在进一步学习数据库故障的处理方面的知识,做为一个数据库维护人员,我即期望遇到所有的数据库出错的案例,以增加自己的经验,但同时又担心遇到这样或那样无法处理的数据库故障而导致数据丢失。

  前几天看到一个文章,是说一个网站管理员在招聘DBA时,提出一个问题:“如果在sql server 日志里发现一个页损坏或是校验和错误应该如何处理?”网站管理员描述,大概有90%的应聘者都会采用一个方案,用DBCC CHECKDB加上其中的一个修复选项,但其中也基本没有人能具体解释DBCC CHECKDB修复的过程或是工作原理及能修复到什么程度。

  借助联机文档以及个人的一些理解和经历,解释一下如何面对这个问题:"当数据库出现页损坏或校验和出错时如何处理?"

  首先,需要先了解DBCC CHECKDB,联机文档url:

  http://technet.microsoft.com/zh-cn/library/ms176064.aspx

  通过联机文档,可以得知有REPAIR_ALLOW_DATA_LOSS | REPAIR_FAST | REPAIR_REBUILD三个修复选项,而提供实际功能的只有REPAIR_ALLOW_DATA_LOSS和REPAIR_REBUILD两个,其中REPAIR_ALLOW_DATA_LOSS 尝试修复报告的所有错误,这些修复可能会导致一些数据丢失;而且REPAIR_REBUILD执行不会丢失数据的修复,包括快速修复(如修复非聚集索引中缺少的行)以及更耗时的修复(如重新生成索引);可见REPAIR_REBUILD是我们期望的。

  当你从sql server log里或是在程序查询数据库或是定期通过DBCC CHECKDB为数据库做体检的时候,出现了页损坏或校验和出错信息时,如:

  ---------------------------------------------------------------------------------------------------------------------------------

  M8928sg , Level 16, State 1, Line 1

  Object ID 2088535921, index ID 0, partition ID 72345201021503994, alloc unit ID 72345201051571606 (type In-row data): Page (1:94299) could not be processed. See other errors for details.

  Msg 8939, Level 16, State 98, Line 1

  Table error: Object ID 2088535921, index ID 0, partition ID 72345201021503994, alloc unit ID 72345201051571606 (type In-row data), page (1:94299). Test (IS_OFF (BUF_IOERR, pBUF->bstat)) failed.

  CHECKDB found 0 allocation errors and 2 consistency errors in table 'yourtable' (object ID 2088535921).

  CHECKDB found 0 allocation errors and 2 consistency errors in database 'yourdb'.

  repair_allow_data_loss is the minimum repair level for the errors found by DBCC CHECKDB (yourdb).

  ---------------------------------------------------------------------------------------------------------------------------------

  现在我们应该如何做?

  1.通过上面的提示,告诉我们:对象 2088535921出错,它是一个表,页面为1:94299

  2.接下来,我们判断损坏的页在堆上还是聚集索引还是非聚集索引,sql server方法为:

  dbcc traceon (3604, -1)

  go

  dbcc page('yourdb', 1, 94299, 3)

  go

  在输出的结果里(会报错,但可以看到页头信息),可以看到

  Metadata: IndexId = n

  如果n是0而表示是堆,1表示是聚集索引,>1是表示非聚集索引

  ps:其实从提示信息的Object ID 2088535921, index ID 0 ,也可以简单判断是堆.

  3.根据上面的第2步,我们知道这个页面是堆,这对我们来讲,不是好消息,因为如果是>1,我们可以删除该非聚集索引,再重建索引,不会丢失数据,而0或1则是元数据受损,这意味着有丢失元数据的可能性。

  那么如何仅仅修复这个数据页呢,这里我们假设该库是full模式,并且有良好的备份策略,有全备和日志备份。

  那么我们可以进行页面级还原操作,步骤如下:

  a.首先进行一次日志备份,如果你不放心,还可以再做一个全备;

  backup log yourdb to disk='D:DBBakyourdb_a.trn'

  b.通过完整备份来恢复该page. (yourdb.bak是一个全备。);

  restore database yourdb page= '1:94299' from disk='D:DBBakyourdb.bak' with norecovery

  c.恢复这个全备之后的差异(假设有差异yourdb.dif),如果没有差异备,直接到d步骤;

  restore database yourdb from disk='d:DBBakyourdb.dif' with norecovery

  d.恢复之后的log备份,可能有多个(假设为yourdb_1.trn,yourdb_2.trn);

  restore log yourdb from disk='d:DBBakyourdb_1.trn' with norecovery

  restore log yourdb from disk='d:DBBakyourdb_2.trn' with norecovery

  restore log yourdb from disk='d:DBBakyourdb_a.trn' with norecovery

  e.做一个最新的日志备;

  backup log yourdb to disk='D:DBBakyourdb_e.trn'

  f.还原最后的(e步骤)日志备份;

  restore log yourdb from disk='d:DBBakyourdb_e.trn' with recovery

  g.结束

  4.经过步骤三之后,我们再来检查一下该表是否还有错,从提示信息Object ID 2088535921里,我们查出表名tbname;

  tbname: select object_name(2088535921)

  然后 dbcc checktable('yourtable')检测,如果没有报错,则表示修复完成

  5.最后,对整个库再做一次dbcc checkdb检查;

  ps:需要注意的是,sql server 的page级恢复在企业版和开发版中,支持联机恢复page数据,在标准版只能脱机修复;

  在dbcc checkdb修复选项里,用repair_rebuild修复数据,联机文档称是不丢失数据,但在某些环境下可能也会丢失数据,不过,我没遇到过:)

  用repair_allow_data_loss 选项时,联机文档称可能会丢失数据,而对于堆或聚集索引的页损坏,sql server 会释放该页面,造成数据的丢失,但repair_allow_data_loss选项有两种情况是不会丢失数据,一种是非聚集索引上的页错误,另外是 lob页数据错误。

  总绍:

  一定要有良好的数据库备份策略,备份重于一切;

  要有异机备份,并且时时同步该备份文件;

  当数据库出现故障时,不要过于心急,冷静分析一下错误;

  如果不能确定如何做,可以借助google,如果你的错误信息里中文的,请翻译成英文后再google,这样搜到解决方案的可能性更大;

  做修复时,一定要再备一次数据库;

  dbcc checkdb的repair_allow_data_loss选项永远是最后的选择。

  结束,如有错误,请指正。

 


推荐阅读
  • 在当前众多持久层框架中,MyBatis(前身为iBatis)凭借其轻量级、易用性和对SQL的直接支持,成为许多开发者的首选。本文将详细探讨MyBatis的核心概念、设计理念及其优势。 ... [详细]
  • 在 Windows 10 中,F1 至 F12 键默认设置为快捷功能键。本文将介绍几种有效方法来禁用这些快捷键,并恢复其标准功能键的作用。请注意,部分笔记本电脑的快捷键可能无法完全关闭。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 计算机网络复习:第五章 网络层控制平面
    本文探讨了网络层的控制平面,包括转发和路由选择的基本原理。转发在数据平面上实现,通过配置路由器中的转发表完成;而路由选择则在控制平面上进行,涉及路由器中路由表的配置与更新。此外,文章还介绍了ICMP协议、两种控制平面的实现方法、路由选择算法及其分类等内容。 ... [详细]
  • 本文将介绍如何使用 Go 语言编写和运行一个简单的“Hello, World!”程序。内容涵盖开发环境配置、代码结构解析及执行步骤。 ... [详细]
  • This guide provides a comprehensive step-by-step approach to successfully installing the MongoDB PHP driver on XAMPP for macOS, ensuring a smooth and efficient setup process. ... [详细]
  • 探讨如何高效使用FastJSON进行JSON数据解析,特别是从复杂嵌套结构中提取特定字段值的方法。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 本文介绍如何利用动态规划算法解决经典的0-1背包问题。通过具体实例和代码实现,详细解释了在给定容量的背包中选择若干物品以最大化总价值的过程。 ... [详细]
  • 本文介绍了在使用Visual Studio 2015进行项目开发时,遇到类向导弹出“异常来自 HRESULT:0x8CE0000B”错误的解决方案。通过具体步骤和实践经验,帮助开发者快速排查并解决问题。 ... [详细]
  • 本文基于刘洪波老师的《英文词根词缀精讲》,深入探讨了多个重要词根词缀的起源及其相关词汇,帮助读者更好地理解和记忆英语单词。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • 数据管理权威指南:《DAMA-DMBOK2 数据管理知识体系》
    本书提供了全面的数据管理职能、术语和最佳实践方法的标准行业解释,构建了数据管理的总体框架,为数据管理的发展奠定了坚实的理论基础。适合各类数据管理专业人士和相关领域的从业人员。 ... [详细]
  • 深入理解 SQL 视图、存储过程与事务
    本文详细介绍了SQL中的视图、存储过程和事务的概念及应用。视图为用户提供了一种灵活的数据查询方式,存储过程则封装了复杂的SQL逻辑,而事务确保了数据库操作的完整性和一致性。 ... [详细]
author-avatar
用释怀来成全悲伤_490_905_560
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有