您好,登錄后才能下訂單哦!
今天就跟大家聊聊有關(guān)使用C#實現(xiàn)一個差異文件備份工具,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。
如歌曲、視頻、文檔,代碼文件等等,如果經(jīng)常增加刪除修改文件,就需要定期備份,最早之前文件都不大的時候我都是手工先全部刪除,然后再全部拷貝,感覺比較保險。后來有了很大的電影文件和很瑣碎的代碼文件之后,這樣搞太折磨人,就學(xué)網(wǎng)上說的用Xcpoy組裝了一個批處理。學(xué)了C#后,感覺還是做一個GUI體驗更好用起來更方便。至于專業(yè)的工具,還真沒怎么試過,有點不放心吧,有好用的倒是可以試試?,F(xiàn)在先自己做一個用著吧。
關(guān)鍵代碼如下:
private async void btnBackUp_Click(object sender, EventArgs e) { string sourceDirectory = txtSource.Text; string targetDirectory = txtTarget.Text; if (sourceDirectory.ToLower() == targetDirectory.ToLower()) { Console.WriteLine("源目錄和備份目錄不能是同一目錄!"); MessageBox.Show("源目錄和備份目錄不能是同一目錄!", "提示", MessageBoxButtons.OK, MessageBoxIcon.Warning); return; } DirectoryInfo diSource = new DirectoryInfo(sourceDirectory); // 源目錄 DirectoryInfo diTarget = new DirectoryInfo(targetDirectory); // 備份目錄 if (diTarget.Name != diSource.Name) diTarget = new DirectoryInfo(Path.Combine(diTarget.FullName, diSource.Name)); // 創(chuàng)建同名目錄 if (!diTarget.Exists) diTarget.Create(); // 如果該目錄已存在,則此方法不執(zhí)行任何操作 btnBackUp.Enabled = false; txtSource.Enabled = false; txtTarget.Enabled = false; lblWork.Text = "備份開始!"; if (await CopyAllAsync(diSource, diTarget)) { lblWork.Text = "備份完成!"; MessageBox.Show("備份完畢!", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information); } else lblWork.Text = "出現(xiàn)錯誤!"; btnBackUp.Enabled = true; txtSource.Enabled = true; txtTarget.Enabled = true; btnBackUp.Focus(); } public async Task<bool> CopyAllAsync(DirectoryInfo source, DirectoryInfo target) { try { foreach (FileInfo fi in source.GetFiles()) // 復(fù)制最新文件 { Console.WriteLine(@"準(zhǔn)備復(fù)制文件 {0}\{1}", target.FullName, fi.Name); // Name不含路徑,僅文件名 FileInfo newfi = new FileInfo(Path.Combine(target.FullName, fi.Name)); if (!newfi.Exists || (newfi.Exists && fi.LastWriteTime > newfi.LastWriteTime)) { Console.WriteLine("正在復(fù)制文件 {0}", newfi.FullName); lblWork.Text = string.Format("正在復(fù)制文件 {0}", newfi.FullName); if (newfi.Exists && newfi.IsReadOnly) newfi.IsReadOnly = false; // 覆蓋或刪除只讀文件會產(chǎn)生異常:對路徑“XXX”的訪問被拒絕 fi.CopyTo(newfi.FullName, true); // Copy each file into it's new directory } } foreach (FileInfo fi2 in target.GetFiles()) // 刪除源目錄沒有而目標(biāo)目錄中有的文件 { FileInfo newfi2 = new FileInfo(Path.Combine(source.FullName, fi2.Name)); if (!newfi2.Exists) { Console.WriteLine("正在刪除文件 {0}", fi2.FullName); lblWork.Text = string.Format("正在刪除文件 {0}", fi2.FullName); if (fi2.IsReadOnly) fi2.IsReadOnly = false; fi2.Delete(); // 沒有權(quán)限(如系統(tǒng)盤需管理員權(quán)限)會產(chǎn)生異常,文件不存在不會產(chǎn)生異常 } } foreach (DirectoryInfo di in source.GetDirectories()) // 復(fù)制目錄(實際上是創(chuàng)建同名目錄,和源目錄的屬性不同步) { Console.WriteLine(" {0} {1}", di.FullName, di.Name); // Name不含路徑,僅本級目錄名 Console.WriteLine(@"準(zhǔn)備創(chuàng)建目錄 {0}\{1}", target.FullName, di.Name); DirectoryInfo newdi = new DirectoryInfo(Path.Combine(target.FullName, di.Name)); if (!newdi.Exists) // 如果CopyAllAsync放在if里的bug: 只要存在同名目錄,則不會進(jìn)行子目錄和子文件的檢查和更新 { Console.WriteLine("正在創(chuàng)建目錄 {0}", newdi.FullName); lblWork.Text = string.Format("正在復(fù)制目錄 {0}", newdi.FullName); DirectoryInfo diTargetSubDir = target.CreateSubdirectory(di.Name); // 創(chuàng)建目錄 Console.WriteLine("完成創(chuàng)建目錄 {0}", diTargetSubDir.FullName); } if (await CopyAllAsync(di, newdi) == false) return false; ; // Copy each subdirectory using recursion } foreach (DirectoryInfo di2 in target.GetDirectories()) // 刪除源目錄沒有而目標(biāo)目錄中有的目錄(及其子目錄和文件) { DirectoryInfo newdi2 = new DirectoryInfo(Path.Combine(source.FullName, di2.Name)); if (!newdi2.Exists) { Console.WriteLine("正在刪除目錄 {0}", di2.FullName); lblWork.Text = string.Format("正在刪除目錄 {0}", di2.FullName); di2.Delete(true); // 只讀的目錄和文件也能刪除,如不使用參數(shù)則異常"目錄不是空的" } } return true; } catch (Exception e) { Console.WriteLine(e.Message); MessageBox.Show(e.Message, "提示", MessageBoxButtons.OK, MessageBoxIcon.Error); return false; } }
注意事項:
// 文件和目錄的創(chuàng)建日期為首次全新復(fù)制時的創(chuàng)建時間
// 文件復(fù)制后修改日期始終保持原先的不變,目錄的修改日期為首次全新復(fù)制時的創(chuàng)建時間(因為本就是新建)
// 單純的覆蓋不會改變修改時間和創(chuàng)建時間
// 文件發(fā)生的屬性變化全新復(fù)制時可以保留(無法通過更新時間判斷文件的屬性變化)
--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
今天測試,又發(fā)現(xiàn)一個bug,真是防不勝防,好在終于找到病根并解決了。
問題出在 if (await CopyAllAsync(diSource, diTarget)) 這個地方,備份開始后,lblWork.Text = "備份開始!"; 結(jié)果發(fā)現(xiàn)標(biāo)簽的設(shè)置并不生效,然后界面很卡,不能拖動窗口。在需要備份更新的文件特別多時感覺更明顯。
原來,設(shè)置控件的Enabled屬性是立即生效,但控件的Text屬性并不是立即生效,就是UI界面不會立即更新,只是將設(shè)置信息加入了windows消息隊列,通常等所在的方法執(zhí)行完畢后才生效,但如果方法中該語句后面還有同類的設(shè)置,就會感覺不到它的生效,其實是生效了,只是先設(shè)為了一個值,然后又立即設(shè)為了另一個值,因為太快了,人眼看不出來。同樣的原因,“正在復(fù)制文件XXX”也不即時顯示正在復(fù)制的文件信息。
然后,界面卡頓,是因為拷貝的時候執(zhí)行緊密運(yùn)算,但是CopyAllAsync(diSource, diTarget)方法并沒有在單獨的線程運(yùn)行,占用了UI線程,導(dǎo)致界面卡頓,改成下面這樣,完美解決:
lblWork.Text = "備份開始!"; bool result = await Task.Run(() => CopyAllAsync(diSource, diTarget)); // 這兒是關(guān)鍵 if (result) // if (await CopyAllAsync(diSource, diTarget)) 開始后界面會卡{ lblWork.Text = "備份完成!"; MessageBox.Show("備份完畢!", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information); } else lblWork.Text = "出現(xiàn)錯誤!";
看完上述內(nèi)容,你們對使用C#實現(xiàn)一個差異文件備份工具有進(jìn)一步的了解嗎?如果還想了解更多知識或者相關(guān)內(nèi)容,請關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。