【SqlServer】日志文件无法收缩的解决方法

前言

正常情况下,不要轻易的收缩日志文件,但是日积月累,日志文件太大,造成硬盘空间不足,不得已去收缩时,却发现收缩不了,那是因为在还原的完整模式下活动事务一直使用日志,所以无法收缩。

可以通过以下步骤:

第一步,查询日志信息(可省略)

sql 复制代码
SELECT 
    name AS 日志文件逻辑名,  --这个就是逻辑名,有的时候逻辑名和我们看到的文件名不一样
    physical_name AS 物理路径,
    size * 8.0 / 1024 AS 当前大小MB
FROM sys.database_files 
WHERE type_desc = 'LOG';

第二步,强制提交事务,并将 恢复模式 改为 简单模式

sql 复制代码
-- 查看数据库中活跃的事务信息,用于检查是否有未提交的事务(可能阻止日志收缩)
-- 语法:DBCC OPENTRAN (数据库名称),会显示最早的活动事务、未分发的事务等信息
DBCC OPENTRAN (databaseName);  


-- 执行检查点操作,强制将内存中已提交的事务日志从缓存写入磁盘
-- 作用:减少日志文件中的活跃内容,为后续收缩做准备(确保已提交数据持久化)
CHECKPOINT;


-- 将数据库恢复模式临时改为简单恢复模式(SIMPLE)
-- 简单模式下,事务日志会自动截断(在检查点后清除不再需要的日志记录),便于收缩日志文件
-- WITH NO_WAIT:表示如果无法立即执行(如存在锁定),不等待直接返回错误(避免长时间阻塞)
ALTER DATABASE databaseName SET RECOVERY SIMPLE WITH NO_WAIT;

第三步 ,开始收缩

sql 复制代码
-- 收缩指定的数据库日志文件
-- 语法:DBCC SHRINKFILE (日志文件逻辑名, 目标大小, 收缩方式)
DBCC SHRINKFILE (
    N'database_log',  -- 日志文件的逻辑名称(需与数据库中定义的一致,可通过sys.database_files查询)
    200,              -- 收缩后的目标大小(单位:MB)
    TRUNCATEONLY      -- 仅截断未使用的日志空间(不移动数据页,效率高,仅适用于日志文件)
);


-- 收缩整个数据库的数据文件和日志文件(通常在收缩日志后补充,进一步释放空间)
-- 语法:DBCC SHRINKDATABASE (数据库名称),会尝试收缩所有文件至最小可能大小(受数据分布影响)
DBCC SHRINKDATABASE(N'数据库名称');  
GO  -- 批处理分隔符,分隔前后语句的执行批次

第四步 恢复数据库的 完整恢复模式

sql 复制代码
-- 将数据库恢复模式改回完整恢复模式(FULL)
-- 完整模式下,日志会记录所有事务细节,支持时间点恢复(生产环境通常需要此模式)
-- WITH NO_WAIT:同上,不等待直接执行或返回错误
ALTER DATABASE databaseName SET RECOVERY FULL WITH NO_WAIT;
相关推荐
倔强的石头_9 小时前
关系数据库替换用金仓:数据迁移过程中的完整性与一致性风险
数据库
Elastic 中国社区官方博客9 小时前
使用 Groq 与 Elasticsearch 进行智能查询
大数据·数据库·人工智能·elasticsearch·搜索引擎·ai·全文检索
穿过锁扣的风10 小时前
一文搞懂 SQL 五大分类:DQL/DML/DDL/DCL/TCL
数据库·microsoft·oracle
l1t10 小时前
DeepSeek总结的SNKV — 无查询处理器的 SQLite 键值存储
数据库·sqlite·kvstore
洛豳枭薰10 小时前
MySQL 梳理
数据库·mysql
九.九10 小时前
CANN 算子生态的底层安全与驱动依赖:固件校验与算子安全边界的强化
大数据·数据库·安全
蓝帆傲亦10 小时前
代码革命!我用Claude Code 3个月完成1年工作量,这些实战经验全给你
jvm·数据库·oracle
亓才孓10 小时前
[JDBC]事务
java·开发语言·数据库
PD我是你的真爱粉10 小时前
FastAPI使用tortoiseORM
数据库·fastapi
剩下了什么18 小时前
MySQL JSON_SET() 函数
数据库·mysql·json