相信很多朋友都經歷過數據庫出問題的情況,我也同樣(見我的上一篇博文:phpmyadmin誤刪表後的恢復過程(心驚膽跳啊) )。如果數據很大或者很重要,那麼恢復起來是相當困難的,所以我們在做一個相對完善的系統時,數據庫的備份/還原功能是必不可少的。本文將在javaEE環境下實現MySQL的自動備份/還原,使用了struts2和hibernate框架,MySQL版本是5.1.16。
下圖展示的是web application的執行流程,
Timer是在一個隨着application啓動而啓動的servlet中初始化,並接受一個名叫‘BACKUP_DEPLY'的參數,它告訴Timer多久去備份一次數據庫(單位爲小時):
1 <servlet> 2 <servlet-name>StartupServlet</servlet-name> 3 <servlet-class>com.nerve.web.servlet.StartupServlet</servlet-class> 4 5 <load-on-startup>1</load-on-startup> 6 7 <init-param> 8 <param-name>BACKUP_DELAY</param-name> 9 <param-value>24</param-value><!--自動備份程序執行間隔,單位爲小時--> 10 </init-param> 11 </servlet>
每次備份數據庫後,會將備份信息保存到數據庫中,爲此我們定義了一個PO,如下:
1 public class Backup{ 2 private int id; 3 private String name; 4 private long size; 5 private Date addDate; 6 7 //setter and getter 8 }
上圖是備份過程中主要用到的類,大概過程爲:
1.當Timer開始執行備份操作時,會調用BackupService(BackupServiceImpl爲其實現類)中的backup()方法
2.BackupService會實例化一個BackWorker(MySQLBackupWorker爲其實現類),並調用它的backup(boolean isRestore)throws Excetion方法來完成備份
3.當BackupWorker的isDone()方法返回true時,則表示備份成功(這時可以通過BackupWorker的getFileName()方法獲取保存的文件名),否則是失敗
BackupWorker接口定義如下:
1 public interface BackupWorker { 2 3 public void backup(boolean isRestore) throws Exception; 4 5 public void reload(String path) throws Exception; 6 7 /** 8 * 是否備份成功 9 * @method name: isDone 10 * @return type: boolean 11 * @return 12 */ 13 public boolean isDone(); 14 15 /** 16 * 獲取備份後文件的名稱(不包括目錄) 17 * @method name: getFileName 18 * @return type: String 19 * @return 20 */ 21 public String getFileName(); 22 23 /** 24 * 獲取備份後文件的大小 25 * @method name: getFileSize 26 * @return type: long 27 * @return 28 */ 29 public long getFileSize(); 30 31 /** 32 * 獲取配置文件中自動備份的時間間隔,單位爲小時 33 * @method name: getHours 34 * @return type: int 35 * @return 36 */ 37 public int getHours(); 38 39 /** 40 * 更新自動備份的時間間隔 41 * @method name: setHours 42 * @return type: void 43 * @param hours 44 */ 45 public void setHours(int hours); 46 47 /** 48 * 刪除備份文件 49 * @method name: delete 50 * @return type: void 51 * @param fileName 52 */ 53 public void delete(String fileName); 54 }
MySQLBackupWorker是MySQL的備份/還原類,實現了BackupWorker接口,在構造函數被執行時,會讀取我們預設的MySQL相關設置(即上面類圖中的MySQLBackup.properties),配置文件中包含如下內容:
(使用時,請將這些配置修改爲本機的實際值,當OS爲Linux是,mysqlpath可以不用設置,前提是 mysqldump 命令有效)
下面是MySQLBackupWorker中在window環境下的備份實現(默認的編碼是utf-8,可以改爲本機值):
1 public void backup(boolean isRe) throws Exception { 2 boolean isWindow = isWindowsOS(); 3 isRestore = isRe; 4 if(isWindow){ 5 this.backupWindow(); 6 }else{ 7 this.backupLinux(); 8 } 9 } 10 11 /** 12 * window下的mysql備份 13 * @method name: backupWindow 14 * @return type: void 15 */ 16 private void backupWindow(){ 17 try { 18 String sqlPath = bkPath + getBackupName(); 19 mkDir(sqlPath); 20 21 StringBuffer sb = new StringBuffer(); 22 sb.append(mysqlPath); 23 sb.append("mysqldump "); 24 sb.append("--opt "); 25 sb.append("-h "); 26 sb.append(host); 27 sb.append(" "); 28 sb.append("--user="); 29 sb.append(loginName); 30 sb.append(" "); 31 sb.append("--password="); 32 sb.append(loginPass); 33 sb.append(" "); 34 sb.append("--lock-all-tables=true "); 35 sb.append("--result-file="); 36 sb.append(sqlPath); 37 sb.append(" "); 38 sb.append("--default-character-set=utf8 "); 39 sb.append(dbName); 40 41 System.out.println(sb.toString()); 42 Runtime cmd = Runtime.getRuntime(); 43 try { 44 Process p = cmd.exec(sb.toString()); 45 int tag = p.waitFor(); 46 System.out.println("result::: "+tag); 47 if(tag == 0) 48 done = true; 49 } catch (IOException e) { 50 e.printStackTrace(); 51 } 52 } catch (Exception e) { 53 e.printStackTrace(); 54 } 55 }
獲取更多源代碼:源代碼(溫馨提示:此鏈接爲淘寶鏈接)
【本文爲小弟原創文章,引用請註明出處,謝謝】