一个小故事

某天,小王正在和HR妹妹闲聊,正HAPPY时,,突然收到系统告警消息,数据库磁盘被剩余空间500M,OMG,不行,磁盘快满了,要是业务要停了,,那就小王只能删库到跑路了,,,

图片 1

 

先检查下,有没有可以删除的不用的文件,结果都是重要的或者拿不准的。先收缩下数据库吧,点击运行。等收缩完成就可以继续去根HR妹妹聊天了。突然电话座机和手机齐鸣,小王心里一种不祥的预感呢?好像这个场景在哪里见过。。不会是数据库阻塞了吧??
手忙脚乱的先接起手机,因为来电显示是某业务部门主管
“小王啊,,现在系统卡死了,全部不动了,是怎么回事啊,你赶紧处理下”,,“恩,好的,我马上检查下”,然后又接起座机,是另外一个部门的主管说报表看不了。慌忙应付完了,赶紧检查数据库执行中的语句。
果然数据库产生大量的阻塞,,连带数据库服务器的操作都变得好慢(是我的心理作用吗?)。正准备先把收缩操作取消了,,电话有同时响起了,,,唉,不管了,先处理问题。然后点击取消。经过漫长的等待,,终于完成了,然后打电话跟各个部门解释,,写事故报告,,悲剧,,今天的午饭都不想吃了。

这个场景是不是很熟悉啊,关于数据库收缩的问题,是我在群里,论坛里,看到新人问过最频繁的问题之一。今天这篇文章对数据库收缩进行有个框架性说明,希望小伙伴在以后遇到相关的

问题时,做到心中有数。

 

 

  开发一个项目时都会有一个蛋疼的问题——写数据库需求文档,然后根据这个文档来建数据库,如果后来需求改了,要改数据库还要改文档,有时忙着忙着就忘改了,导致文档是过期的。那么我们自己写个脚本在数据库运行直接生产数据字典,这样只要改数据库就行了。目前在网上搜了下,发现sqlServer只有2005的生成工具,没有08的,存储过程倒是有,不过下载运行一遍到处是坑,写的也太差了,于是对脚本进行改进。

mysql 官方客户端  MySQL-Workbench

关于收缩的建议

不到万不得已,千万不要收缩数据库。收缩数据库影响极大:

1.收缩数据库对数据库的影响极大,产生大量日志和碎片,而且会锁表。如果你的库当前正在被使用,收缩不下去非常正常。
2.收缩数据库一定要手工来做的,而且是在维护窗口期做的事。
3.尽量使用语句来执行,可以提示错误

下面的文章详细介绍:
.

 

     

下载链接 http://dev.mysql.com/downloads/workbench/

收缩的正确姿势

在不得不收缩的时候,参考下面的步骤

1.找到数据库中最大的几个表,重建所有索引。首先尝试指定Truncate
Only收缩方式.它只是移除文件尾部的空闲空间,并不重新组织已经使用的数据页。

DBCC SHRINKDATABASE (AdventureWorks2012, TRUNCATEONLY);  

2 最后才考虑,不带选项的收缩。收缩不要一次性全部收缩。
可以每次收缩2G左右。不要把空间可用空间全部收缩了,可以剩余一部分比如4G。收缩完后,记得重建索引.

补充:

还有一种办法就是新建文件组,使用CREATE INDEX … WITH(DROP_EXISTING =
ON)ON语法将所有相关的的表和索引移动到新文件组。然后收缩旧的文件组。

 

3.可在进程中的任一点停止 DBCC SHRINKDATABASE
操作,任何已完成的工作都将保留。

  1. 不能在备份数据库时收缩数据库

 

-- =============================================
-- Author:        <marlon>
-- Create date: <2017-11-13>
-- Description:    <生成数据库字典>
-- =============================================

BEGIN
    DECLARE @TableName nvarchar(35),@htmls varchar(8000)
    DECLARE @字段名称 VARCHAR(200)
    DECLARE @类型  VARCHAR(200)
    DECLARE @长度 VARCHAR(200)
    DECLARE @数值精度 VARCHAR(200)
    DECLARE @小数位数 VARCHAR(200)
    DECLARE @默认值 VARCHAR(200)
    DECLARE @允许为空 VARCHAR(200)
    DECLARE @外键 VARCHAR(200)
    DECLARE @主键 VARCHAR(200)
    DECLARE @描述 VARCHAR(200)

    SET NOCOUNT ON;

    DECLARE Tbls CURSOR
    FOR
        Select distinct Table_name
        FROM INFORMATION_SCHEMA.COLUMNS
        order by Table_name
    OPEN Tbls
        PRINT '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">'
        PRINT '<html xmlns="http://www.w3.org/1999/xhtml">'
        PRINT '    <head>'
        PRINT '        <title>SqlServer数据字典</title>'
        PRINT '        <style type="text/css">'
        PRINT '            body{margin:0; font:11pt "arial", "微软雅黑"; cursor:default;}'
        PRINT '            .titleHead{margin:0 auto;text-align:center;}'
        PRINT '            .tableBox{margin:10px auto; padding:0px; width:1000px; height:auto; background:#FBF5E3; border:1px solid #45360A}'
        PRINT '            .tableBox h3 {font-size:12pt; height:30px; line-height:30px; background:#45360A; padding:0px 0px 0px 15px; color:#FFF; margin:0px; text-align:left }'
        PRINT '            .tableBox table {width:1000px; padding:0px }'
        PRINT '            .tableBox th {height:25px; border-top:1px solid #FFF; border-left:1px solid #FFF; background:#F7EBC8; border-right:1px solid #E0C889; border-bottom:1px solid #E0C889 }'
        PRINT '            .tableBox td {height:25px; padding-left:10px; border-top:1px solid #FFF; border-left:1px solid #FFF; border-right:1px solid #E0C889; border-bottom:1px solid #E0C889 }'
        PRINT '            .tableBox td {text-align:center}'
        PRINT '        </style>'
        PRINT '    </head>'
        PRINT '    <body>'
        PRINT '       <div class="titleHead">'
        PRINT '          <h1>'+DB_NAME()+'数据字典</h1>'
        PRINT '       </div>'
    FETCH NEXT FROM Tbls INTO @TableName
    WHILE @@FETCH_STATUS = 0
    BEGIN
  set @htmls = ''
        Select  @htmls = '        <h3>' + @TableName + ' : '+ CAST(Value as varchar(1000)) + '</h3>'
        FROM sys.extended_properties AS A
        WHERE A.major_id = OBJECT_ID(@TableName)
        and minor_id = 0

        if @htmls is null or DATALENGTH (@htmls)=0
            begin
            set @htmls = '        <h3>' + @TableName + '</h3>'
            end
        PRINT '        <div class="tableBox">'
        PRINT @htmls
        PRINT '            <table cellspacing="0">'
        PRINT '                <tr>'
        PRINT '                    <th>字段名称</th>'
        PRINT '                    <th>类型</th>'
        PRINT '                    <th>长度</th>'
        PRINT '                    <th>数值精度</th>'
        PRINT '                    <th>小数位数</th>'
        PRINT '                    <th>默认值</th>'
        PRINT '                    <th>是否允许为空</th>'
        PRINT '                    <th>是否外键</th>'
        PRINT '                    <th>是否主键</th>'
        PRINT '                    <th>描述</th>'
        PRINT '                </tr>'

        DECLARE TRows CURSOR
        FOR
            SELECT
            '                    <td>' + CAST(clmns.name AS VARCHAR(35)) + '</td>',
            '                    <td>' + CAST(udt.name AS CHAR(15)) + '</td>' ,
            '                    <td>' + CAST(CAST(CASE WHEN typ.name IN (N'nchar', N'nvarchar') AND clmns.max_length <> -1 THEN clmns.max_length/2 ELSE clmns.max_length END AS INT) AS VARCHAR(20)) + '</td>',
            '                    <td>' + CAST(CAST(clmns.precision AS INT) AS VARCHAR(20)) + '</td>',
            '                    <td>' + CAST(CAST(clmns.scale AS INT) AS VARCHAR(20)) + '</td>',
            '                    <td>' + isnull(CAST(cnstr.definition AS VARCHAR(20)),'') + '</td>',
            '                    <td>' + (case when clmns.is_nullable > 0 then '是' else '否' end) + '</td>' ,
            '                    <td>' + (case when clmns.is_computed > 0 then '是' else '否' end) + '</td>' ,
            '                    <td>' + (case when clmns.is_identity > 0 then '是' else '否' end) + '</td>' ,
            '                    <td>' + ISNULL(CAST(exprop.value AS VARCHAR(500)),'') + '</td>'
            FROM sys.tables AS tbl 
            INNER JOIN sys.all_columns AS clmns ON clmns.object_id=tbl.object_id
            LEFT OUTER JOIN sys.indexes AS idx ON idx.object_id = clmns.object_id AND 1 =idx.is_primary_key
            LEFT OUTER JOIN sys.index_columns AS idxcol ON idxcol.index_id = idx.index_id AND idxcol.column_id = clmns.column_id AND idxcol.object_id = clmns.object_id AND 0 = idxcol.is_included_column
            LEFT OUTER JOIN sys.types AS udt ON udt.user_type_id = clmns.user_type_id
            LEFT OUTER JOIN sys.types AS typ ON typ.user_type_id = clmns.system_type_id AND typ.user_type_id = typ.system_type_id
            LEFT JOIN sys.default_constraints AS cnstr ON cnstr.object_id=clmns.default_object_id
            LEFT OUTER JOIN sys.extended_properties exprop ON exprop.major_id = clmns.object_id AND exprop.minor_id = clmns.column_id AND exprop.name = 'MS_Description'
            WHERE (tbl.name = @TableName) 
            ORDER BY clmns.column_id ASC        
        OPEN TRows
        FETCH NEXT FROM TRows INTO @字段名称,@类型,@长度,@数值精度,@小数位数,@默认值,@允许为空,@外键,@主键,@描述
        WHILE @@FETCH_STATUS = 0
        BEGIN
            PRINT '                <tr>'
            PRINT @字段名称
            PRINT @类型
            PRINT @长度
            PRINT @数值精度
            PRINT @小数位数
            PRINT @默认值
            PRINT @允许为空
            PRINT @外键
            PRINT @主键
            PRINT @描述
            PRINT '                </tr>'
            FETCH NEXT FROM TRows INTO @字段名称,@类型,@长度,@数值精度,@小数位数,@默认值,@允许为空,@外键,@主键,@描述
        END
        CLOSE TRows
        DEALLOCATE TRows

        PRINT '            </table>'
        PRINT '        </div>'
    FETCH NEXT FROM Tbls INTO @TableName
    END
        PRINT '    </body>'
        PRINT '</html>'
    CLOSE Tbls
    DEALLOCATE Tbls
END

具体安装步骤就不写了,直接一直下一步就可以了。

 可能需要收缩的场景

1.你删除了大量数据,而且数据不太可能增长。

2.要移除某个文件时,你需要先清空数据文件。

 

  注意事项:

下面说一下基础操作:

总结

那我们处理磁盘空间不足的最好的办法是什么呢?最好的办法是在最初规划时,预估好未来一年或者二年的数据增长。给磁盘划分足够的空间。设置好数据库的初始大小,并且将自动增长使用固定量增长。

 

  1、以上脚本直接在SQL SERVER运行,以文本格式显示,设置如下:

 登录成功后,界面如下所示。其中,区域1显示的是数据库服务器中已经创建的数据库列表。区域2是关于数据库的操作列表。区域三是sql的编辑器和执行环境,区域4是执行结果的列表

  图片 2

图片 3

  红框的不要勾选

在sql的编辑器中输入测试语句,如图所示,选择执行(或者使用快捷键ctrl+enter)。执行成功后,查询结果会显示在下面的列表中。

发表评论

电子邮件地址不会被公开。 必填项已用*标注