伍佰目录 短网址
  当前位置:海洋目录网 » 站长资讯 » 站长资讯 » 文章详细 订阅RssFeed

误删除MySQL数据库表的ibd文件怎么办

来源:本站原创 浏览:96次 时间:2022-11-12

很多年前,学习oracle10g的技术时,看过盖国强老师写的一篇文章,在linux系统中,oracle数据库存活的时候,用系统命令rm删除一个数据文件,这个时候找到删除文件的fd文件句柄,就可以将删除的文件找回,当时感觉特别的神奇,竟然还有这种骚操作。

既然oracle数据库可以恢复删除的数据文件,那在MySQL数据库中能不能玩这样的骚操作呢,下面就是我做的模拟测试,一起来看看吧。

模拟测试表

[root@localhost] 10:18:14 [(none)]>use testdb;Reading table information for completion of table and column namesYou can turn off this feature to get a quicker startup with -ADatabase changed[root@localhost] 10:18:19 [testdb]>[root@localhost] 10:18:20 [testdb]>show tables;+------------------+| Tables_in_testdb |+------------------+| t_test           || test1            |+------------------+2 rows in set (0.00 sec)[root@localhost] 10:18:24 [testdb]>select * from test1;+----+-------+-------+| id | name | name2 |+----+-------+-------+|  1 | test1 | test1 ||  2 | test  | test  |+----+-------+-------+2 rows in set (0.01 sec)

查找Mysql数据库进程IP

[root@mysql ~]# ps -ef|grep -i mysqld|grep -i basedir|grep -v grepmysql      7988   7159  1 10:17 pts/0    00:00:02 /data/mysql-5.7.26/bin/mysqld --defaults-file=/data/mysql/conf/3306/my.cnf --basedir=/data/mysql-5.7.26 --datadir=/data/mysql/data/3306 --plugin-dir=/data/mysql-5.7.26/lib/plugin --log-error=/data/mysql/log/3306/error.log --open-files-limit=65535 --pid-file=/data/mysql/run/3306/mysql.pid --socket=/data/mysql/run/3306/mysql.sock --port=3306

查看Mysql数据库进程所用使用的文件句柄

[root@mysql ~]# cd /proc/7988/fd[root@mysql fd]# ll总用量 0lr-x------ 1 mysql mysql 64 9月  18 10:18 0 -> /dev/nulllrwx------ 1 mysql mysql 64 9月  18 10:18 42 -> /data/mysql/data/3306/mysql/tables_priv.MYDlrwx------ 1 mysql mysql 64 9月  18 10:18 43 -> /data/mysql/data/3306/mysql/columns_priv.MYIlrwx------ 1 mysql mysql 64 9月  18 10:18 44 -> /data/mysql/data/3306/mysql/columns_priv.MYDlrwx------ 1 mysql mysql 64 9月  18 10:18 45 -> /data/mysql/data/3306/mysql/procs_priv.MYIlrwx------ 1 mysql mysql 64 9月  18 10:18 46 -> /data/mysql/data/3306/mysql/procs_priv.MYDlrwx------ 1 mysql mysql 64 9月  18 10:18 47 -> /data/mysql/data/3306/mysql/servers.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 48 -> /data/mysql/data/3306/mysql/slave_master_info.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 49 -> /data/mysql/data/3306/mysql/slave_relay_log_info.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 5 -> /tmp/ibTITlZK (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 50 -> /data/mysql/data/3306/mysql/slave_worker_info.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 51 -> /data/mysql/data/3306/mysql/event.MYIlrwx------ 1 mysql mysql 64 9月  18 10:18 52 -> /data/mysql/data/3306/mysql/event.MYDlrwx------ 1 mysql mysql 64 9月  18 10:21 53 -> socket:[39748]lrwx------ 1 mysql mysql 64 9月  18 10:21 54 -> /data/mysql/data/3306/query_rewrite/rewrite_rules.ibdlrwx------ 1 mysql mysql 64 9月  18 10:21 55 -> /data/mysql/data/3306/testdb/test1.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 6 -> /tmp/ib3ojR5h (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 7 -> /tmp/ib9mkncP (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 8 -> /tmp/ibrfgk1T (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 9 -> /data/mysql/log/3306/redo/ib_logfile1

模拟删除业务表ibd文件

[mysql@mysql testdb]$ ls -ltotal 728-rw-rw-r--. 1 mysql mysql 485385 Jul 14 11:11 a.log-rw-r-----. 1 mysql mysql     67 Jul 13 17:20 db.opt-rw-r-----  1 mysql mysql   8618 Sep 18 10:31 test1.frm-rw-r-----  1 mysql mysql  98304 Sep 18 10:32 test1.ibd-rw-r-----. 1 mysql mysql   8618 Sep  9 15:23 t_test.frm-rw-r-----. 1 mysql mysql 131072 Sep  9 15:25 t_test.ibd[mysql@mysql testdb]$[mysql@mysql testdb]$ rm test1.ibd[mysql@mysql testdb]$ ls -ltotal 632-rw-rw-r--. 1 mysql mysql 485385 Jul 14 11:11 a.log-rw-r-----. 1 mysql mysql     67 Jul 13 17:20 db.opt-rw-r-----  1 mysql mysql   8618 Sep 18 10:31 test1.frm-rw-r-----. 1 mysql mysql   8618 Sep  9 15:23 t_test.frm-rw-r-----. 1 mysql mysql 131072 Sep  9 15:25 t_test.ibd

如果你在生产环境,不小心做了这个操作,估计这个时候已经背后一身冷汗了。


再次查看Mysql数据库进程所用使用的文件句柄

[root@mysql fd]# ls -llrwx------ 1 mysql mysql 64 9月  18 10:18 46 -> /data/mysql/data/3306/mysql/procs_priv.MYDlrwx------ 1 mysql mysql 64 9月  18 10:18 47 -> /data/mysql/data/3306/mysql/servers.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 48 -> /data/mysql/data/3306/mysql/slave_master_info.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 49 -> /data/mysql/data/3306/mysql/slave_relay_log_info.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 5 -> /tmp/ibTITlZK (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 50 -> /data/mysql/data/3306/mysql/slave_worker_info.ibdlrwx------ 1 mysql mysql 64 9月  18 10:18 51 -> /data/mysql/data/3306/mysql/event.MYIlrwx------ 1 mysql mysql 64 9月  18 10:18 52 -> /data/mysql/data/3306/mysql/event.MYDlrwx------ 1 mysql mysql 64 9月  18 10:21 53 -> socket:[39748]lrwx------ 1 mysql mysql 64 9月  18 10:21 54 -> /data/mysql/data/3306/query_rewrite/rewrite_rules.ibdlrwx------ 1 mysql mysql 64 9月  18 10:21 55 -> /data/mysql/data/3306/testdb/test1.ibd (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 6 -> /tmp/ib3ojR5h (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 7 -> /tmp/ib9mkncP (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 8 -> /tmp/ibrfgk1T (deleted)lrwx------ 1 mysql mysql 64 9月  18 10:18 9 -> /data/mysql/log/3306/redo/ib_logfile1[root@mysql fd]# 

这个表在一定的时间范围内,还是可以进行DML操作

[root@localhost] 10:49:54 [testdb]>update test1 set name2='dsljfld' where id=2;Query OK, 1 row affected (0.00 sec)Rows matched: 1  Changed: 1  Warnings: 0[root@localhost] 10:50:12 [testdb]>select * from test1;+----+-------+---------+| id | name  | name2   |+----+-------+---------+|  1 | test1 | test1   ||  2 | test  | dsljfld |+----+-------+---------+2 rows in set (0.00 sec)[root@localhost] 10:50:36 [testdb]>insert into test1 values(3,'dsf','sfsf');Query OK, 1 row affected (0.01 sec)[root@localhost] 10:50:45 [testdb]>select * from test1;+----+-------+---------+| id | name  | name2   |+----+-------+---------+|  1 | test1 | test1   ||  2 | test  | dsljfld ||  3 | dsf   | sfsf    |+----+-------+---------+

为了保险起见,建议将这个表进行lock操作,只能read操作

[root@localhost] 10:58:17 [testdb]>lock tables test1 read;Query OK, 0 rows affected (0.00 sec)[root@localhost] 10:59:07 [testdb]>[root@localhost] 10:59:09 [testdb]>insert into test1 values(4,'esdsf','dfesfsf');ERROR 1099 (HY000): Table 'test1' was locked with a READ lock and can't be updated

你会发现,刚才rm test1.ibd文件,在这里还可以看到,这个时候,我们要做的是,赶紧将这个文件拷贝回去。

[mysql@mysql fd]$ cp /proc/9015/fd/54 /data/mysql/data/3306/testdb/test1.ibd[mysql@mysql testdb]$ ls -ltotal 728-rw-rw-r--. 1 mysql mysql 485385 Jul 14 11:11 a.log-rw-r-----. 1 mysql mysql     67 Jul 13 17:20 db.opt-rw-r-----  1 mysql mysql   8618 Sep 18 10:31 test1.frm-rw-r-----  1 mysql mysql  98304 Sep 18 11:01 test1.ibd-rw-r-----. 1 mysql mysql   8618 Sep  9 15:23 t_test.frm-rw-r-----. 1 mysql mysql 131072 Sep  9 15:25 t_test.ibd

可以看到文件已经完全找回去了。这个时候,可以将表的read lock释放了

[root@localhost] 11:02:56 [testdb]>unlock tables;Query OK, 0 rows affected (0.00 sec)[root@localhost] 11:03:25 [testdb]>insert into test1 values(4,'esdsf','dfesfsf');Query OK, 1 row affected (0.00 sec)

此时你重启MySQL数据库都是没有任何问题的。

总结
为什么能通过这种方式,恢复rm掉的ibd文件呢,主要是因为当我们用rm命令删除的时候,Mysql数据库进程还在持有被删除的ibd文件的句柄,也就是在/proc/{mysql_pid}/pd目录下可以找到,如果你这个时候,重启了Mysql数据库实例,Mysql进程就会释放掉删除文件的句柄,你就真的访问不到被删除的文件了,这种情况下,要恢复被删除的表,就会很费时和费力了。

  推荐站点

  • At-lib分类目录At-lib分类目录

    At-lib网站分类目录汇集全国所有高质量网站,是中国权威的中文网站分类目录,给站长提供免费网址目录提交收录和推荐最新最全的优秀网站大全是名站导航之家

    www.at-lib.cn
  • 中国链接目录中国链接目录

    中国链接目录简称链接目录,是收录优秀网站和淘宝网店的网站分类目录,为您提供优质的网址导航服务,也是网店进行收录推广,站长免费推广网站、加快百度收录、增加友情链接和网站外链的平台。

    www.cnlink.org
  • 35目录网35目录网

    35目录免费收录各类优秀网站,全力打造互动式网站目录,提供网站分类目录检索,关键字搜索功能。欢迎您向35目录推荐、提交优秀网站。

    www.35mulu.com
  • 就要爱网站目录就要爱网站目录

    就要爱网站目录,按主题和类别列出网站。所有提交的网站都经过人工审查,确保质量和无垃圾邮件的结果。

    www.912219.com
  • 伍佰目录伍佰目录

    伍佰网站目录免费收录各类优秀网站,全力打造互动式网站目录,提供网站分类目录检索,关键字搜索功能。欢迎您向伍佰目录推荐、提交优秀网站。

    www.wbwb.net