Mysql ibdata 丟失或損壞如何通過frm&ibd 恢復數據

 mysql存儲在磁盤中,各種天災人禍都會導致數據丟失。大公司的時候我們常常需要做好數據冷熱備,對於小公司來說要做好所有數據備份需要支出大量的成本,很多公司也是不現實的。萬一還沒有做好備份,數據被誤刪除了,或者ibdata損壞了怎麼辦呢?別擔心,只要有部分的frm、ibd存在就可以恢復部分數據。

注意:
一、這個是對innodb的數據恢復。myisam不需要這麼麻煩,只要數據文件存在直接複製過去就可以。
二、大家的mysql數據庫必須是按表存放數據的,默認不是,但是大家生產肯定是按分表設置的吧,如果不是,不好意思,這個方法不能恢復你的數據。my.ini的設置爲 innodb_file_per_table          = 1。
 
 
1、找回表結構,如果表結構沒有丟失直接到下一步
      a、先創建一個數據庫,這個數據庫必須是沒有表和任何操作的。
      b、創建一個表結構,和要恢復的表名是一樣的。表裏的字段無所謂。一定要是innodb引擎的。CREATE TABLE `weibo_qq0`(  `weiboid` bigint(20)) ENGINE=InnoDB DEFAULT CHARSET=utf8;
      c、關閉mysql, service mysqld stop;
      d、用需要恢復的frm文件覆蓋剛新建的frm文件;   
      e、修改my.ini 裏  innodb_force_recovery=1 , 如果不成修改爲 2,3,4,5,6。
      f、 啓動mysql,service mysqld start;show create table weibo_qq0 就能夠看到表結構信息了。
 
2、找回數據。記得上面把 innodb_force_recovery改掉了,需要註釋掉,不然恢復模式不好操作。 這裏有個關鍵的問題,就是innodb裏的任何數據操作都是一個日誌的記錄點。也就是如果我們需要數據恢復,必須把之前的表的數據的日誌記錄點添加到一致。 
      a、建立一個數據庫,根據上面導出的創建表的sql執行創建表。
      b、找到記錄點。先要把當前數據庫的表空間廢棄掉,使當前ibd的數據文件和frm分離。  ALTER TABLE weibo_qq0 DISCARD TABLESPACE;
      c、把之前要恢復的 .ibd文件複製到新的表結構文件夾下。 使當前的ibd 和frm發生關係。ALTER TABLE weibo_qq0  IMPORT TABLESPACE; 結果不出意外肯定會報錯。就和我們開展數據開始說的那樣,數據記錄點不一致。我們看看之前ibd記錄的點在什麼位置。開始執行 import tablespace,報錯  ERROR 1030 (HY000): Got error -1 from storage engine。找到mysql的錯誤日誌,InnoDB: Error: tablespace id in file '.\test\weibo_qq0.ibd' is 112, but in the InnoDB  InnoDB: data dictionary it is 1.  因爲  weibo_qq0 之前的記錄點在112,當前的表只創建一次,所以記錄點是1.
      d、那怎麼從1記錄到112。for ($1=1; $i<=111; $1++) {CREATE TABLE t# (id int) ENGINE=InnoDB;}  也許很奇怪,爲什麼是循環111,不是112。因爲在a執行創建表結構的時候已經記錄增加了一次。
      e、修改表結構 alter table weibo_qq0  discard tablespace;使當前的表結構和ibd脫離關係。複製.ibd到當前的目錄結構。
      f、使原來數據的ibd和當前frm建立關係。 ALTER TABLE product IMPORT TABLESPACE; 這個時候沒有錯誤,說明已經建立好了。但是查詢數據還是查不出來。
      g、相比這裏大家已經知道爲什麼了,這個模式也不是說改了數據庫就可以在生產環境使用。更改 innodb_force_recovery=1 , 如果不成修改爲 2,3,4,5,6。直到可以 查詢出數據爲止,然後dump出來。數據就備份出來了。
      h、把所有數據導出後,在新的數據庫導入。所有數據就生成了。
 
 
      擴展問題,很多時候我們是分表表結構怎麼批量操作,提高速度呢。用循環!循環把表的空間廢棄掉。
for i in `seq 0 111`; do mysql -uroot -P33061 -h127.0.0.1 -Dtestdd  -e "CREATE TABLE inv_crawl_weibo_qq$i (id bigint(20) NOT NULL AUTO_INCREMENT,PRIMARY KEY (id)) ENGINE=innodb "; done
ALTER TABLE  inv_crawl_weibo_qq0  DISCARD TABLESPACE;
從備份數據把ibd複製cp到dd數據庫下,注意複製過來的文件權限。
循環導入表空間。
ALTER TABLE inv_crawl_weibo_qq0   IMPORT TABLESPACE;
沒有報錯就導入成功了
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章