MYSQL表碎片整理

注:在修改MySQL数据库数据表引擎发现有很多碎片

一、查看表碎片大小

(1)查看某个表的碎片大小
 SHOW TABLE STATUS LIKE '表名';   结果中’Data_free’列的值就是碎片大小
 
 2)列出所有已经产生碎片的表
 select table_schema db, table_name, data_free, engine     
from information_schema.tables
where table_schema not in ('information_schema', 'mysql')  and data_free > 0;

 二、 清除表碎片

    MyISAM表:optimize table 表名

    InnoDB表:alter table 表名 engine=InnoDB

    注:清除碎片操作会暂时锁表,数据量越大,耗费的时间越长


  三、MYSQL表碎片
    #!/bin/sh  
    mysql_user=root   
    mysql_pass=123123  
    time_log=/opt/time
    databases=/opt/databases  
    /usr/bin/mysql -u$mysql_user -p$mysql_pass -e "show databases" | grep -v "Database" > /opt/databases  
    sed -i "s/information_schema//" $databases  
    sed -i "s/mysql//" $databases  
    sed -i "s/test//" $databases  
    databases1=$(cat /opt/databases)  
    for i in $databases1  
    do  
    echo "database $i starting"
    tables=$(/usr/bin/mysql $i -u$mysql_user -p$mysql_pass -e "show tables" | grep -v "Tables" > /opt/$i)  
    tablelist=$(cat /opt/$i)  
    echo "optimize database $i starting" >> $time_log  
    echo "$i start at $(date  +[%Y/%m/%d/%H:%M:%S])" >> $time_log  
    for list in $tablelist  
    do  
    echo $list  
    /usr/bin/mysql $i -u$mysql_user -p$mysql_pass -e "optimize table $list"
    done  
    echo "$i end   at $(date  +[%Y/%m/%d/%H:%M:%S])" >> $time_log  
    echo >> $time_log  
    done


四:说明
经常会使用VARCHAR、TEXT、BLOB等可变长度的文本数据类型。不过,当我们使用这些数据类型之后,我们就不得不做一些额外的工作——MySQL数据表碎片整理。
每当MySQL从你的列表中删除了一行内容,该段空间就会被留空。而在一段时间内的大量删除操作,会使这种留空的空间变得比存储列表内容所使用的空间更大。
当MySQL对数据进行扫描时,它扫描的对象实际是列表的容量需求上限,也就是数据被写入的区域中处于峰值位置的部分。如果进行新的插入操作,MySQL将尝试利用这些留空的区域,但仍然无法将其彻底占用。

1.或者查看某个表所占空间,以及碎片大小。

select table_name,engine,table_rows,data_length+index_length length,DATA_FREE from information_schema.tables where TABLE_SCHEMA='db002';
或者

select table_name,engine,table_rows,data_length+index_length length,DATA_FREE from information_schema.tables where TABLE_SCHEMA='db002' and data_free !=0;


2、整理表的碎片
optimize table aaa_safe,aaa_user,t_platform_user,t_user;

提示该表不支持 optimize,但是下边有显示OK.其实已经执行成功了。5.6.X的版本,其实已经支持Innodb了。
整理完毕


针对以前版本:
大致意思是说innodb的数据库不支持optimize,可以用
ALTER TABLE table.name ENGINE='InnoDB';
该方法会对旧表以复制的方式新建一个新表,然后删除旧表。虽然这个过程是安全的,但是在进行操作时还是先进行备份为好
也可以在启动数据库的时候指定--skip-new或者--safe-mode选项来支持optimize功能
[root@wg ~]# service mysql start --skip-new

OPTIMIZE TABLE只对MyISAM, BDB和InnoDB表起作用。

 

对于MyISAM表,OPTIMIZE TABLE按如下方式操作:

1.     如果表已经删除或分解了行,则修复表。

2.     如果未对索引页进行分类,则进行分类。

3.        如果表的统计数据没有更新(并且通过对索引进行分类不能实现修复),则进行更新。

 

对于BDB表,OPTIMIZE TABLE目前被映射到ANALYZE TABLE上。

对于InnoDB表,OPTIMIZE TABLE被映射到ALTER TABLE上,这会重建表。重建操作能更新索引统计数据并释放成簇索引中的未使用的空间。

备注:
1.MySQL官方建议不要经常(每小时或每天)进行碎片整理,一般根据实际情况,只需要每周或者每月整理一次即可。
2.OPTIMIZE TABLE只对MyISAM,BDB和InnoDB表起作用,尤其是MyISAM表的作用最为明显。此外,并不是所有表都需要进行碎片整理,一般只需要对包含上述可变长度的文本数据类型的表进行整理即可。
3.在OPTIMIZE TABLE运行过程中,MySQL会锁定表。
4.默认情况下,直接对InnoDB引擎的数据表使用OPTIMIZE TABLE,可能会显示「 Table does not support optimize, doing recreate + analyze instead」的提示信息。这个时候,我们可以用mysqld --skip-new或者mysqld --safe-mode命令来重启MySQL,以便于让其他引擎支持OPTIMIZE TABLE。

 整理碎片过程会锁边,尽量放在业务低峰期做操作

1、myisam存储引擎回收碎片optimize table aaa_safe,aaa_user,t_platform_user,t_user;

2、innodb存储引擎回收碎片alter table t engine=innodb;

1.MySQL官方建议不要经常(每小时或每天)进行碎片整理,一般根据实际情况,只需要每周或者每月整理一次即可。

2.OPTIMIZE TABLE运行过程中,MySQL会锁定表。4.默认情况下,直接对InnoDB引擎的数据表使用OPTIMIZE TABLE或

脚本回收innodb表碎片

#!/bin/bash

DB=test

USER=root

PASSWD=root123

HOST=192.168.2.202

MYSQL_BIN=/usr/local/mysql/bin

D_ENGINE=InnoDB

$MYSQL_BIN/mysql -h$HOST -u$USER -p$PASSWD $DB -e "select TABLE_NAME from information_schema.TABLES where TABLE_SCHEMA='"$DB"' "';" | grep -v "TABLE_NAME" >tables.txt

for t_name in `cat tables.txt`

do

echo "Starting table $t_name......"

sleep 1

$MYSQL_BIN/mysql -h$HOST -u$USER -p$PASSWD $DB -e "alter table $t_name engine='"$D_ENGINE"'"

if [ $? -eq 0 ]

then

echo "shrinktable $t_name ended." >>con_table.log

sleep 1

else

echo "shrinkfailed!" >> con_table.log

fi

done
五、mysql 碎片整理存储过程与shell

1、存储过程

    DELIMITER $$
        DROP PROCEDURE IF EXISTS `mysql`.`sp_optimize_tables`$$
        CREATE PROCEDURE `mysql`.`sp_optimize_tables`(
         IN db_name varchar(255))
        BEGIN
          -- Created by david yeung 20080128.
          -- To optimize all the tables in exact database.
          declare cnt int default 0;
          declare i int default 0;
          select count(*) as total from information_schema.tables where table_schema = db_name into cnt;
          while i < cnt
          do
            -- Get the table's exact name.
            set @stmt = concat('select table_name from information_schema.tables where table_schema = ''',db_name,''' order by table_name asc limit ',i,',1 into @tb_name');
            prepare s1 from @stmt;
            execute s1;
            drop prepare s1;
            set @stmt = '';
            set @stmt = concat('optimize table ',db_name,'.',@tb_name);
            prepare s1 from @stmt;
            execute s1;
            drop prepare s1;
            set @stmt = '';
            set i = i + 1;
          end while;
          -- Refresh tables.
          flush tables;
        END$$
        DELIMITER ;

2、调用示例:

    mysql> use mysql
    Database changed
    mysql> call sp_optimize_tables('david_test');
    +------------------------------+----------+----------+----------+
    | Table | Op | Msg_type | Msg_text |
    +------------------------------+----------+----------+----------+
    | david_test.test1 | optimize | status | OK |
    +------------------------------+----------+----------+----------+
    1 row in set (0.26 sec)

3、在SHELL中调用

    #!/bin/sh
    #
    # Created by david yeung.
    # Optimize all the table one by one.
    #
    cd /usr/local/mysql/bin
    ./mysql -uoptimize_user -poptimize -e "use \"$1\";call mysql.sp_optimize_tables(\"$1\")"
    不过要注意给optimize_user的select,insert权限。

2、碎片整理的shell脚本

    #!/bin/sh
        mysql_user=root
        mysql_pass=123123
        time_log=/opt/time
        databases=/opt/databases
        /usr/bin/mysql -u$mysql_user -p$mysql_pass -e "show databases" | grep -v "Database" >/opt/databases
        sed -i "s/information_schema//" $databases
        sed -i "s/mysql//" $databases
        sed -i "s/test//" $databases
        databases1=$(cat /opt/databases)
        for i in $databases1
        do
        echo "database $i starting"
        tables=$(/usr/bin/mysql $i -u$mysql_user -p$mysql_pass -e "show tables" | grep -v "Tables" >/opt/$i)
        tablelist=$(cat /opt/$i)
        echo "optimize database $i starting" >> $time_log
        echo "$i start at $(date +[%Y/%m/%d/%H:%M:%S])" >> $time_log
        for list in $tablelist
        do
        echo $list
        /usr/bin/mysql $i -u$mysql_user -p$mysql_pass -e "optimize table $list"
        done
        echo "$i end at $(date +[%Y/%m/%d/%H:%M:%S])" >> $time_log
        echo >> $time_log
        done


 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值