日期:2014-05-18  浏览次数:20476 次

sql server日志问题
我一直对数据库日志理解不深,想请教一下各位大虾:
问题1:

所有的数据库操作都会产生日志么?我用下面代码测试发现日志大小却没变化

create table t (t0 nvarchar(50),t1 nvarchar(50),t2 nvarchar(50),t3 nvarchar(50),t4 nvarchar(50))

select cast([size] as decimal(9,2))*8.00/1024.00,* from sysfiles

declare @i int
declare @s nvarchar(50)

set @s='sdfdsfjskdjkdsfjdksfjkdsfjksdjfksdfjdk'
set @i=10000
while @i>0
begin
insert into t (t0,t1,t2,t3,t4) values (@s,@s,@s,@s,@s)
set @i=@i-1
end

select cast([size] as decimal(9,2))*8.00/1024.00,* from sysfiles
drop table t

当我删除该表的所有数据时,日志文件增加了,怎么回事呢?

问题2: 
由于日志的疯狂增长,不得不控制日志,因为数据库空间收费是(数据空间+日志空间)* 单价的,由于日志的增加导致客户严重不满,所以我想控制日志大小,比如说100M,当日志文件超过100M时,自动备份以下日志,然后截断日志超过100M的部分,这样的话,客户日志到达一定的时候变化就不大了,但我必须保留最新的日志,就是说截取是从前朝后还是从后朝前的,要是每次都把最新的截断了,害怕出现问题时难以恢复。(不能分离数据库删除日志文件,因为这样会导致整个空间波动较大,客户会以为数据丢失了)

我的情况是这样的:一个中心数据库,20个子库,想通过中心数据库对子库进行管理,如创建/修改/删除/日志截断/收费等等。。

各位高手们帮忙了,不知你们对日志是怎么处理的

------解决方案--------------------
问题2:

清除日志:

 
DECLARE @LogicalFileName sysname,
@MaxMinutes INT,
@NewSize INT
USE szwzcheck -- 要操作的数据库名
SELECT @LogicalFileName = 'szwzcheck_Log', -- 日志文件名
@MaxMinutes = 10, -- Limit on time allowed to wrap log.
@NewSize = 20 -- 你想设定的日志文件的大小(M)
-- Setup / initialize
DECLARE @OriginalSize int
SELECT @OriginalSize = size 
FROM sysfiles
WHERE name = @LogicalFileName
SELECT 'Original Size of ' + db_name() + ' LOG is ' + 
CONVERT(VARCHAR(30),@OriginalSize) + ' 8K pages or ' + 
CONVERT(VARCHAR(30),(@OriginalSize*8/1024)) + 'MB'
FROM sysfiles
WHERE name = @LogicalFileName
CREATE TABLE DummyTrans
(DummyColumn char (8000) not null)
DECLARE @Counter INT,
@StartTime DATETIME,
@TruncLog VARCHAR(255)
SELECT @StartTime = GETDATE(),
@TruncLog = 'BACKUP LOG ' + db_name() + ' WITH TRUNCATE_ONLY'
DBCC SHRINKFILE (@LogicalFileName, @NewSize)
EXEC (@TruncLog)
-- Wrap the log if necessary.
WHILE @MaxMinutes > DATEDIFF (mi, @StartTime, GETDATE()) -- time 
AND @OriginalSize = (SELECT size FROM sysfiles WHERE name = 
@LogicalFileName)
AND (@OriginalSize * 8 /1024) > @NewSize
BEGIN -- Outer loop.
SELECT @Counter = 0
WHILE ((@Counter < @OriginalSize / 16) AND (@Counter < 50000))
BEGIN -- update
INSERT DummyTrans VALUES ('Fill Log')
DELETE DummyTrans
SELECT @Counter = @Counter + 1
END
EXEC (@TruncLog)
END
SELECT 'Final Size of ' + db_name() + ' LOG is ' +
CONVERT(VARCHAR(30),size) + ' 8K pages or ' + 
CONVERT(VARCHAR(30),(size*8/1024)) + 'MB'
FROM sysfiles 
WHERE name = @LogicalFileName
DROP TABLE DummyTrans
SET NOCOUNT OFF
 
把szwzcheck换成你数据库的名字即可,在查询分析器里面运行。 
有全角的空格(为了显示好看),你自己把他换一下. 


收缩日志:

企业管理器--所有任务--收缩数据库--文件--选日志文件收缩

------解决方案--------------------
--最好备份日志,以后可通过日志恢复数据。。。 
以下为日志处理方法 
一般不建议做第4,6两步 
第4步不安全,有可能损坏数据库或丢失数据 
第6步如果日志达到上限,则以后的数据库处理会失败,在清理日志后才能恢复. 
--*/ 

--下面的所有库名都指你要处理的数据库的库名 

1.清空日志 
DUMP TRANSACTION 库名 WITH NO_LOG

2.截断事务日志: 
BACKUP LOG 库名 WITH NO_LOG 

3.收缩数据库文件(如果不压缩,数据库的文件不会减小 
企业管理器--右键你要压缩的数据库--所有任务--收缩数据库--收缩文件 
--选择日志文件--在收缩方式里选择收缩至XXM,这里会给出一个允许收缩到的最小M数,直接输入这个数,确定就可以了 
--选择数据文件--在收缩方式里选择收缩至XXM,这里会给出一个允许收缩到的最小M数,直接