1 前言
長(cháng)期以來(lái),廣大程序員為到底是使用Client/Server,還是使用Browser/Server結構爭論不休,在這些爭論當中,C/S結構的程序可維護性差,布置困難,升級不方便,維護成本高就是一個(gè)相當重要的因素。有很多企業(yè)用戶(hù)就是因為這個(gè)原因而放棄使用C/S。然而當一個(gè)應用必須要使用C/S結構才能很好的實(shí)現其功能的時(shí)候,我們該如何解決客戶(hù)端的部署與自動(dòng)升級問(wèn)題?部署很簡(jiǎn)單,只要點(diǎn)擊安裝程序即可,難的在于每當有新版本發(fā)布時(shí),能夠實(shí)現自動(dòng)升級[3]?,F在好了,我們的目標很簡(jiǎn)單,我們希望開(kāi)發(fā)一個(gè)與具體應用無(wú)關(guān)的能夠復用的自動(dòng)升級系統。下面我為大家提供了一套可復用的用C#編寫(xiě)的自動(dòng)升級系統。
2 實(shí)現軟件的自動(dòng)升級存在的困難
第一,為了查找遠程服務(wù)器上的更新,應用程序必須有查詢(xún)網(wǎng)絡(luò )的途徑,這需要網(wǎng)絡(luò )編程、簡(jiǎn)單的應用程序與服務(wù)器通訊的協(xié)議。
第二是下載。下載看起來(lái)不需要考慮聯(lián)網(wǎng)的問(wèn)題,但要考慮下載用戶(hù)請求的文件,以及在沒(méi)有用戶(hù)同意時(shí)下載大文件。友好的自動(dòng)更新應用程序將使用剩余的帶寬下載更新。這聽(tīng)起來(lái)簡(jiǎn)單,但卻是一個(gè)技術(shù)難題,幸運的是已經(jīng)有了解決方法。
第三個(gè)考慮因素是使用新版應用程序更換原應用程序的過(guò)程。這個(gè)問(wèn)題比較有趣,因為它要求代碼運行時(shí)將自己從系統刪除,有多種辦法可以實(shí)現該功能[5],本文程序主要通過(guò)比較新舊版本的日期號來(lái)實(shí)現替換新版本應用程序的功能。
3 實(shí)現軟件自動(dòng)在線(xiàn)升級的原理
寫(xiě)兩個(gè)程序,一個(gè)是主程序;一個(gè)是升級程序;所有升級任務(wù)都由升級程序完成。
1.啟動(dòng)升級程序,升級程序連接到網(wǎng)站,下載新的主程序(當然還包括支持的庫文件、XML配置文檔等)到臨時(shí)文件夾;
2.升級程序獲取服務(wù)器端XML配置文件中新版本程序的更新日期或版本號或文件大??;
3.升級程序獲取原有客戶(hù)端應用程序的最近一次更新日期或版本號或文件大小,兩者進(jìn)行比較;如果發(fā)現升級程序的日期大于原有程序的最新日期,則提示用戶(hù)是否升級;或者是采用將現有版本與最新版本作比較,發(fā)現最新的則提示用戶(hù)是否升級;也有人用其它屬性如文件大小進(jìn)行比較,發(fā)現升級程序的文件大小大于舊版本的程序的大小則提示用戶(hù)升級。本文主要采用比較新舊版本更新日期號來(lái)提示用戶(hù)升級。
4.如果用戶(hù)選擇升級,則獲取下載文件列表,開(kāi)始進(jìn)行批量下載文檔;
5.升級程序檢測舊的主程序是否活動(dòng),若活動(dòng)則關(guān)閉舊的主程序;
6.刪除舊的主程序,拷貝臨時(shí)文件夾中的文件到相應的位置;
7.檢查主程序的狀態(tài),若狀態(tài)為活動(dòng)的,則啟動(dòng)新的主程序;
8.關(guān)閉升級程序,升級完成[4]。
4 用C#實(shí)現在線(xiàn)升級的關(guān)鍵步驟
這里我主要使用日期信息來(lái)檢測是否需要下載升級版本。
4.1 準備一個(gè)XML配置文件
名稱(chēng)為AutoUpdater.xml,作用是作為一個(gè)升級用的模板,顯示需要升級的信息。
<?xml version="1.0"?> //xml版本號
<AutoUpdater>
<URLAddres URL="
http://192.168.198.113/vbroker/log/"/>//升級文件所在服務(wù)器端的網(wǎng)址
<UpdateInfo>
<UpdateTime Date = "2005-02-02"/> //升級文件的更新日期
<Version Num = "1.0.0.1"/> //升級文件的版本號
</UpdateInfo>
<UpdateFileList> //升級文件列表
<UpdateFile FileName = "aa.txt"/> //共有三個(gè)文件需升級
<UpdateFile FileName = "VB40.rar"/>
<UpdateFile FileName = "VB4-1.CAB"/>
</UpdateFileList>
<RestartApp>
<ReStart Allow = "Yes"/> //允許重新啟動(dòng)應用程序
<AppName Name = "TIMS.exe"/> //啟動(dòng)的應用程序名
</RestartApp>
</AutoUpdater>
從以上XML文檔中可以得知升級文檔所在服務(wù)器端的地址、升級文檔的更新日期、需要升級的文件列表,其中共有三個(gè)文件需升級:aa.txt、VB40.rar、VB4-1.CAB。以及是否允許重新啟動(dòng)應用程序和重新啟動(dòng)的應用程序名。
4.2 獲取客戶(hù)端應用程序及服務(wù)器端升級程序的最近一次更新日期
通過(guò)GetTheLastUpdateTime()函數來(lái)實(shí)現。
private string GetTheLastUpdateTime(string Dir)
{
string LastUpdateTime = "";
string AutoUpdaterFileName = Dir + @"\AutoUpdater.xml";
if(!File.Exists(AutoUpdaterFileName))
return LastUpdateTime;
//打開(kāi)xml文件
FileStream myFile = new FileStream(AutoUpdaterFileName,FileMode.Open);
//xml文件閱讀器
XmlTextReader xml = new XmlTextReader(myFile);
while(xml.Read())
{
if(xml.Name == "UpdateTime")
{
//獲取升級文檔的最后一次更新日期
LastUpdateTime = xml.GetAttribute("Date");
break;
}
}
xml.Close();
myFile.Close();
return LastUpdateTime;
}
通過(guò)XmlTextReader打開(kāi)XML文檔,讀取更新時(shí)間從而獲取Date對應的值,即服務(wù)器端升級文件的最近一次更新時(shí)間。
函數調用實(shí)現:
//獲取客戶(hù)端指定路徑下的應用程序最近一次更新時(shí)間
string thePreUpdateDate = GetTheLastUpdateTime(Application.StartupPath);
Application.StartupPath指客戶(hù)端應用程序所在的路徑。
//獲得從服務(wù)器端已下載文檔的最近一次更新日期
string theLastsUpdateDate = GetTheLastUpdateTime(theFolder.FullName);
theFolder.FullName指在升級文檔下載到客戶(hù)機上的臨時(shí)文件夾所在的路徑。
4.3 比較日期
客戶(hù)端應用程序最近一次更新日期與服務(wù)器端升級程序的最近一次更新日期進(jìn)行比較。
//獲得已下載文檔的最近一次更新日期
string theLastsUpdateDate = GetTheLastUpdateTime(theFolder.FullName);
if(thePreUpdateDate != "")
{
//如果客戶(hù)端將升級的應用程序的更新日期大于服務(wù)器端升級的應用程序的更新日期
if(Convert.ToDateTime(thePreUpdateDate)>=Convert.ToDateTime(theLastsUpdateDate))
{
MessageBox.Show("當前軟件已經(jīng)是最新的,無(wú)需更新!","系統提示",MessageBoxButtons.OK,MessageBoxIcon.Information);
this.Close();
}
}
this.labDownFile.Text = "下載更新文件";
this.labFileName.Refresh();
this.btnCancel.Enabled = true;
this.progressBar.Position = 0;
this.progressBarTotal.Position = 0;
this.progressBarTotal.Refresh();
this.progressBar.Refresh();
//通過(guò)動(dòng)態(tài)數組獲取下載文件的列表
ArrayList List = GetDownFileList(GetTheUpdateURL(),theFolder.FullName);
string[] urls = new string[List.Count];
List.CopyTo(urls, 0);
將客戶(hù)端升級的應用程序的日期與服務(wù)器端下載的應用程序日期進(jìn)行比較,如果前者大于后者,則不更新;如果前者小于后者,則通過(guò)動(dòng)態(tài)數組獲取下載文件的列表,開(kāi)始下載文件。
通過(guò)BatchDownload()函數來(lái)實(shí)現。升級程序檢測舊的主程序是否活動(dòng),若活動(dòng)則關(guān)閉舊的主程序;刪除舊的主程序,拷貝臨時(shí)文件夾中的文件到相應的位置;檢查主程序的狀態(tài),若狀態(tài)為活動(dòng)的,則啟動(dòng)新的主程序。
private void BatchDownload(object data)
{
this.Invoke(this.activeStateChanger, new object[]{true, false});
try
{
DownloadInstructions instructions = (DownloadInstructions) data;
//批量下載
using(BatchDownloader bDL = new BatchDownloader())
{
bDL.CurrentProgressChanged += new DownloadProgressHandler(this.SingleProgressChanged);
bDL.StateChanged += new DownloadProgressHandler(this.StateChanged);
bDL.FileChanged += new DownloadProgressHandler(bDL_FileChanged);
bDL.TotalProgressChanged += new DownloadProgressHandler(bDL_TotalProgressChanged);
bDL.Download(instructions.URLs, instructions.Destination, (ManualResetEvent) this.cancelEvent);
}
}
catch(Exception ex)
{
ShowErrorMessage(ex);
}
this.Invoke(this.activeStateChanger, new object[]{false, false});
this.labFileName.Text = "";
//更新程序
if(this._Update)
{
//關(guān)閉原有的應用程序
this.labDownFile.Text = "正在關(guān)閉程序....";
System.Diagnostics.Process[]proc=System.Diagnostics.Process.GetProcessesByName("TIMS");
//關(guān)閉原有應用程序的所有進(jìn)程
foreach(System.Diagnostics.Process pro in proc)
{
pro.Kill();
}
DirectoryInfo theFolder=new DirectoryInfo(Path.GetTempPath()+"JurassicUpdate");
if(theFolder.Exists)
{
foreach(FileInfo theFile in theFolder.GetFiles())
{
//如果臨時(shí)文件夾下存在與應用程序所在目錄下的文件同名的文件,則刪除應用程序目錄下的文件
if(File.Exists(Application.StartupPath + \\"+Path.GetFileName(theFile.FullName)))
File.Delete(Application.StartupPath + "\\"+Path.GetFileName(theFile.FullName));
//將臨時(shí)文件夾的文件移到應用程序所在的目錄下
File.Move(theFile.FullName,Application.StartupPath + \\"+Path.GetFileName(theFile.FullName));
}
}
//啟動(dòng)安裝程序
this.labDownFile.Text = "正在啟動(dòng)程序....";
System.Diagnostics.Process.Start(Application.StartupPath + "\\" + "TIMS.exe");
this.Close();
}
}
這段程序是實(shí)現在線(xiàn)升級的關(guān)鍵代碼,步驟有點(diǎn)復雜:首先用Invoke方法同步調用狀態(tài)改變進(jìn)程,然后調用動(dòng)態(tài)鏈接庫中批量下載(BatchDownloader.cs)類(lèi)啟動(dòng)批量下載,接著(zhù)判斷原有的主應用程序有沒(méi)有關(guān)閉,如果沒(méi)關(guān)閉,則用Process.Kill()來(lái)關(guān)閉主程序。接下來(lái),判斷臨時(shí)文件夾下(即下載升級程序所在的目錄)是否存在與應用程序所在目錄下的文件同名的文件,如果存在同名文件,則刪除應用程序目錄下的文件,然后將臨時(shí)文件夾的文件移到應用程序所在的目錄下。最后重新啟動(dòng)主應用程序。這樣更新就完成了。