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

SQLServer数据库不收缩

SQLSe

我将大胆猜测并说您的数据库设置为使用完全恢复。您从未根据上图备份过日志(更不用说完整的日志了)。如果是这种情况,那么您的日志文件会越来越大,并且无法清除,因为您没有进行备份。

如果您有房间,请进行完整备份,然后进行日志备份并再次尝试收缩。应该工作正常。

最坏的情况(我的意思是最坏的,特别是如果这是一个生产数据库)是将数据库设置为 SIMPLE 恢复,收缩然后进行备份。即使您没有足够的空间正常收缩,这也应该有效。

此时请确保为数据库设置定期备份。


你能告诉我们日志与数据文件的空闲百分比是多少吗?您也可以查看 sys.databases 中的 log_reuse_wait_desc。

SELECT name, log_reuse_wait_desc FROM sys.databases

最后但同样重要的是,您是否尝试过TRUNCATeonLY收缩选项?

DBCC SHRINKFILE (Adventureworks2008R2_Log, TRUNCATeonLY)



这可能有很多原因。我将提及所有可能的原因,然后定义解决问题的解决方案。


  1. 如您的DBCC输出所述,您没有足够的可用磁盘空间来缩小数据库文件。换句话说。您尝试将页面从数据库文件移动到新创建的文件。在此期间,页面存在两次。之后 SQL Server 将删除旧的数据库文件并将新创建的文件用于生产。您可能需要增加磁盘空间或按照下面提到的解决方案进行操作。

  2. 防止收缩的另一点可能是数据库锁(这不是你的情况)。如果数据库在 DDL 更改期间被锁定,它将无法收缩数据库文件。如果您使用向导缩小文件,您可能会不时遇到超时消息。如果您将它作为脚本运行,它通常会一直等待直到锁被释放,然后收缩文件。

最简单的解决方案是计划!检查您的数据库并尝试评估每天将插入多少数据,确定增长率。默认情况下,SQL Server 会将文件增大 10%。这对于只有 100MB 的数据库来说还不错,因为增长仅为 10MB。但是,如果您的数据库变得越来越大,增长将耗费大量资源和时间(100GB -> 10 GB,1TB -> 100GB 等)。以 MB 为单位定义增长因子。它不应该太小(例如 1MB),因为它会在大事务中给您的磁盘带来压力,并会减慢您的操作,但也不能太大而导致巨大的写入操作(例如 10GB)。

另外定义可能升级的数据库的增长限制(例如日志数据库)。如果达到限制,事务可能会失败,但不会对日志记录数据库造成太大影响。但是,如果您的日志记录数据库用完了所有可用的磁盘空间,则可能会导致生产数据库上的事务中止/回滚。

如果您遇到问题,即您的磁盘空间太小,无法再次容纳已用磁盘空间(例如:140GB 文件,40GB 可用空间,意味着 100GB 用于重组 -> 您至少需要 100GB 可用空间用于移动操作收缩的时间),您仍然可以通过使用TRUNCATeonLY标志来收回一些空间。

DBCC SHRINKFILE (1,TRUNCATeonLY)

这将切断所有保留在 file_id = 1 的数据库文件末尾的空数据页。回到我们的 140GB 示例:如果您的 140GB 数据库文件在 120GB 的位置保存最后写入的页面,您将获得最后 20GB 几乎立即恢复。这个过程非常快,因为它只是将文件结束标记移动到 120GB 的位置,并将通知您的操作系统有关可用空间的信息。

您可以使用此查询快速了解数据库文件:

SELECT *
FROM sys.sysfiles

如果您有足够的可用磁盘空间(或者在您TRUNCATeonLY对每个文件运行之后),您应该能够真正清理您的数据库文件并真正缩小它们。

这可以通过使用以下语句来实现:

DBCC SHRINKFILE (1);

如果您有一个经常写入或经常运行更大事务的数据库表,我建议定义一个目标限制。这取决于您使用该数据库的经验。通常为 5-10%(取决于数据库的大小),但最小值应设置为 100-1000MB。以我们的 140GB 数据库文件(40GB 空)为例,目标大小应设置为 110GB(100GB 已用 + 10% 保留)。

您可以使用以下语句实现此目的:

DBCC SHRINKFILE (1,112640); -- 110GB in MB

但请注意,这可能会运行很长时间(取决于您的文件大小和 I/O 子系统)。由于您将在 I/O 子系统上创建更大的工作负载,因此我建议在低负载时间(可能是一夜之间)运行此类脚本。

如果你真的想清理你的数据库文件,检查所有索引和REBUILD/或REORGANIZE所有具有高碎片的索引是个好主意。

如果你在开发机器上或者真的愿意冒险,你可以运行这个语句。它将遍历数据库中的每个索引并重建它。

DeclARE @sql nvarchar(max), @sch nvarchar(max), @obj nvarchar(max), @ind nvarchar(max)
DeclARE cur CURSOR FOR
SELECT s.name, o.name, i.name
FROM sys.objects as o
INNER JOIN sys.schemas as s
ON o.schema_id = s.schema_id
INNER JOIN sys.indexes as i
ON o.object_id = i.object_id
WHERE i.type > 0
OPEN cur
FETCH NEXT FROM cur INTO @sch, @obj, @ind
WHILE @@FETCH_STATUS = 0 BEGIN
SET @sql = N'ALTER INDEX ['+@ind+'] ON ['+@sch+'].['+@obj+'] '
+ N'REBUILD PARTITION = ALL '
+ N'WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, SORT_IN_TEMPDB = OFF, '
+ N'OnLINE= OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON)'
PRINT(@sql) EXEC (@sql)
FETCH NEXT FROM cur INTO @sch, @obj, @ind
END
CLOSE cur
DEALLOCATE cur

在该过程之后,您的数据库文件可能会再次增长。但是您可以将其缩小到更小的文件大小,因为每个索引中的可用空间都被删除了。

收缩数据库文件时要小心。SQL Server 提前分配文件空间是有充分理由的。磁盘分配很昂贵,尽可能少做是个好主意。如果您有一个非常敏捷的数据库,每天产生几 GB 的开销并在一段时间后释放它,那么总是将文件缩小到最小值并不是一件好事。您浪费光盘能量来增大文件,做您的工作并再次浪费光盘能量以再次缩小文件。在这种情况下,请注意浪费一点磁盘空间来改善您的数据操作。

希望这个冗长的总结能帮助您解决问题,甚至帮助您制定数据库计划。



在 SQL Server 2017 上,查询尝试包含一些系统表。在“WHERE i.type > 0”下面添加“AND o.is_ms_shipped = 0”解决了这个问题





推荐阅读
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • Windows服务与数据库交互问题解析
    本文探讨了在Windows 10(64位)环境下开发的Windows服务,旨在定期向本地MS SQL Server (v.11)插入记录。尽管服务已成功安装并运行,但记录并未正确插入。我们将详细分析可能的原因及解决方案。 ... [详细]
  • DNN Community 和 Professional 版本的主要差异
    本文详细解析了 DotNetNuke (DNN) 的两种主要版本:Community 和 Professional。通过对比两者的功能和附加组件,帮助用户选择最适合其需求的版本。 ... [详细]
  • MySQL 数据库迁移指南:从本地到远程及磁盘间迁移
    本文详细介绍了如何在不同场景下进行 MySQL 数据库的迁移,包括从一个硬盘迁移到另一个硬盘、从一台计算机迁移到另一台计算机,以及解决迁移过程中可能遇到的问题。 ... [详细]
  • 本文详细探讨了JDBC(Java数据库连接)的内部机制,重点分析其作为服务提供者接口(SPI)框架的应用。通过类图和代码示例,展示了JDBC如何注册驱动程序、建立数据库连接以及执行SQL查询的过程。 ... [详细]
  • PostgreSQL 10 离线安装指南
    本文详细介绍了如何在无法联网的服务器上进行 PostgreSQL 10 的离线安装,并涵盖了从下载安装包到配置远程访问的完整步骤。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • CentOS7源码编译安装MySQL5.6
    2019独角兽企业重金招聘Python工程师标准一、先在cmake官网下个最新的cmake源码包cmake官网:https:www.cmake.org如此时最新 ... [详细]
  • 本文详细介绍了如何通过多种编程语言(如PHP、JSP)实现网站与MySQL数据库的连接,包括创建数据库、表的基本操作,以及数据的读取和写入方法。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 在使用 DataGridView 时,如果在当前单元格中输入内容但光标未移开,点击保存按钮后,输入的内容可能无法保存。只有当光标离开单元格后,才能成功保存数据。本文将探讨如何通过调用 DataGridView 的内置方法解决此问题。 ... [详细]
  • 使用Vultr云服务器和Namesilo域名搭建个人网站
    本文详细介绍了如何通过Vultr云服务器和Namesilo域名搭建一个功能齐全的个人网站,包括购买、配置服务器以及绑定域名的具体步骤。文章还提供了详细的命令行操作指南,帮助读者顺利完成建站过程。 ... [详细]
  • 根据最新发布的《互联网人才趋势报告》,尽管大量IT从业者已转向Python开发,但随着人工智能和大数据领域的迅猛发展,仍存在巨大的人才缺口。本文将详细介绍如何使用Python编写一个简单的爬虫程序,并提供完整的代码示例。 ... [详细]
  • 本文详细介绍了 MySQL 中 LAST_INSERT_ID() 函数的使用方法及其工作原理,包括如何获取最后一个插入记录的自增 ID、多行插入时的行为以及在不同客户端环境下的表现。 ... [详细]
  • 本文探讨了MariaDB在当前数据库市场中的地位和挑战,分析其可能面临的困境,并提出了对未来发展的几点看法。 ... [详细]
author-avatar
暮迟_MCz_P
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有