軟體自動更新的原理

來源:互聯網
上載者:User
需要自動更新的時候,用戶端軟體提交更新要求給固定的網址,然後網址返回最新版本的版本號碼。用戶端軟體和自己的版本號碼對比,如果探索服務器的版本和自己的版本不一樣,那麼就請求下載升級補丁。伺服器響應請求,提供補丁下載;下載成功後,手動或自動安裝一下就可以了。

    安裝補丁的時候,用戶端的主程式一般是要關閉的,除非採用動態卸載、調入組件的技術,比如網路遊戲,就是盡量讓玩家邊玩遊戲邊更新。

    如果更新的是主程式,那麼就必須關閉主程式了。可是主程式關閉了之後,誰來調用安裝補丁呢?為瞭解決這個問題,人們把主程式和自動更新程式分開來做。當需要校正版本的時候,主程式調用自動更新程式。自動更新程式如果發現主程式需要更新,在下載了升級補丁之後,就會要求關閉主程式。主程式關閉之後,自動更新程式調用升級補丁進行安裝,安裝完成後再重新啟動主程式。自動更新程式自動結束,完成更新任務。這和你剛才關閉電腦後再插網卡的原理是一模一樣的。其實生活中有很多現象可以借鑒到軟體設計中。

   人們嫌這樣的升級仍然太麻煩,後來出現了B/S結構,大家像抓住了救命稻草一樣的追捧至今,希望能從這些勞動中解脫出來。B/S結構就是瀏覽器/伺服器結構,它所有的功能,都是在瀏覽器裡完成。程式升級的時候,只需要在服務端升級一下就可以了,瀏覽器再次訪問伺服器的時候,得到的就是新版本。你可以把它理解成一個純粹的網站。是不是站長更新的時候,你開啟網站就看到最新的頁面?

   但是B/S結構有天生的弱點,因為它是基於網頁瀏覽的,所以安全性是首要的難題,你能瀏覽的東西,原則上別人也能瀏覽。而且出於安全考慮,B/S訪問本地資源,比如串口,必須藉助於其他技術實現。B/S也不是萬能的,必須考慮實際應用。

附錄:Winform的自動更新

winform程式相對web程式而言,功能更強大,編程更方便,但軟體更新卻相當麻煩,要到用戶端一台一台地升級,面對這個實際問題,在最近的一個小項目中,本人設計了一個通過軟體實現自動升級技術方案,彌補了這一缺陷,有較好的參考價值。 

一、升級的好處。 
長期以來,廣大程式員為到底是使用Client/Server,還是使用Browser/Server結構爭論不休,在這些爭論當中,C/S結構的程式的可維護性差,布置困難,升級不方便,維護成本高就是一個相當重要的因素,也是那些B/S的支援者們將Client/Server結構打入地獄的一個重要原因。 

現在好了,我們就在最新的基於Microsoft 的 WinForm上用WebServices來實現軟體的自動升級功能。 

二、升級的技術原理。 
升級的原理有好幾個,首先無非是將現有版本與最新版本作比較,發現最新的則提示使用者是否升級。當然也有人用其它屬性比較的,例如:檔案大小。 或者更新日期。 
而實現的方法呢?在VB時代,我使用的是XmlHTTP+INet控制項。用XmlHTTP擷取資訊,用INET傳輸升級檔案,而用一個簡單的BAT檔案來實現升級。 

Public Sub CheckUpdate() 
On Error Resume Next 
Dim b As Boolean 
Dim XmlHttp As Object 
Set XmlHttp = CreateObject("Microsoft.XMLHttp" 
XmlHttp.Open "GET", "Http://mu.5inet.net/MuAdmin/update.xml", False 
XmlHttp.Send 

Dim vs As String 
vs = XmlHttp.responseText 
If Err.Number > 0 Then 
Exit Sub 
End If 

Dim Xml As Object 
Set Xml = CreateObject("Microsoft.XmlDom" 
Xml.LoadXml vs 
Dim Version As String 
Dim downAddr As String 
Dim FSize As Long 
Dim fInfo As String 
Version = Xml.DocumentElement.ChildNodes(0).Text 
downAddr = Xml.DocumentElement.ChildNodes(1).Text 
FSize = CLng(Xml.DocumentElement.ChildNodes(2).Text) 
fInfo = Xml.DocumentElement.ChildNodes(3).Text 
Set Xml = Nothing 
Set XmlHttp = Nothing 

Dim Major As Long 
Dim Minor As Long 
Dim Revision As Long 
Dim C() As String 
C = Split(Version, "." 
Major = CLng(C(0)) 
Minor = CLng(C(1)) 
Revision = CLng(C(2)) 

If Major > App.Major Then 
b = True 
ElseIf Minor > App.Minor Then 
b = True 
ElseIf Revision > App.Revision Then 
b = True 
Else 
b = False 
End If 
If (b) Then 
Dim result As VbMsgBoxResult 
result = MsgBox("發現程式新版本。目前的版本為:" & App.Major & "." & App.Minor & "." & App.Revision & ",目前最新版本為:" & Version & ",是否進行更新?", vbQuestion Or vbYesNo, "自動更新" 
If result = vbYes Then 
Dim frm As New Update 
frm.DownloadAddress = downAddr 
frm.size = FSize 
frm.InfoPage = fInfo 
frm.Version = Version 
frm.Show vbModal 
End If 
End If 
End Sub 

而BAT檔案有個特性,是可以刪除自己本身。下面是BAT檔案的內容. 
@echo off 
echo 
echo echo 歡迎使用無垠奇蹟管理器升級嚮導。 
echo 本次升級版本為:1.1.0。 
echo 請按任意鍵開始升級無垠奇蹟管理器... echo 
echo 
pause 
del SQLSrvBrowser.Exe 
ren ~update.tmp SQLSrvBrowser.Exe 
echo 升級成功,按任意鍵重新啟動應用程式。 
pause 
start http://mu.5inet.net/  
start SQLSrvBrowser.Exe 
del update.bat 

三、在.Net時代的實現。 
在.Net時代,我們就有了更多的選擇,可以使用WebRequest,也可以使用WebServices。在這裡我們將用WebServices來實現軟體的自動升級。 

實現原理:在WebServices中實現一個GetVer的WebMethod方法,其作用是擷取當前的最新版本。 
  然後將現在版本與最新版本比較,如果有新版本,則進行升級。 

  步驟: 
    1、準備一個XML檔案 (Update.xml)。 
<?xml version="1.0" encoding="utf-8" ?> 
<product> 
<version>1.0.1818.42821</version> 
<description>修正一些Bug</description> 
<filelist count="4" sourcepath="./update/"> 
<item name="City.xml" size=""> 
<value /> 
</item> 
<item name="CustomerApplication.exe" size=""> 
<value /> 
</item> 
<item name="Interop.SHDocVw.dll" size=""> 
<value /> 
</item> 
<item name="Citys.xml" size=""> 
<value /> 
</item> 
</filelist> 
</product> 
作用是作為一個升級用的模板。 
    2、WebServices的GetVer方法。 

[WebMethod(Description="取得更新版本"] 
public string GetVer() 

XmlDocument doc = new XmlDocument(); 
doc.Load(Server.MapPath("update.xml"); 
XmlElement root = doc.DocumentElement; 
return root.SelectSingleNode("version".InnerText; 

     3、WebServices的GetUpdateData方法。 
[WebMethod(Description="線上更新軟體"] 
[SoapHeader("sHeader"] 
public System.Xml.XmlDocument GetUpdateData() 

//驗證使用者是否登陸 
if(sHeader==null) 
return null; 
if(!DataProvider.GetInstance.CheckLogin(sHeader.Username,sHeader.Password)) 
return null; 
//取得更新的xml模板內容 
XmlDocument doc = new XmlDocument(); 
doc.Load(Server.MapPath("update.xml"); 
XmlElement root = doc.DocumentElement; 
//看看有幾個檔案需要更新 
XmlNode updateNode = root.SelectSingleNode("filelist"; 
string path = updateNode.Attributes["sourcepath"].Value; 
int count = int.Parse(updateNode.Attributes["count"].Value); 
//將xml中的value用實際內容替換 
for(int i=0;i<count;i++) 

XmlNode itemNode = updateNode.ChildNodes[i]; 
string fileName = path + itemNode.Attributes["name"].Value; 
FileStream fs = File.OpenRead(Server.MapPath(fileName)); 
itemNode.Attributes["size"].Value = fs.Length.ToString(); 
BinaryReader br = new BinaryReader(fs); 
//這裡是檔案的實際內容,使用了Base64String編碼 
itemNode.SelectSingleNode("value".InnerText = Convert.ToBase64String(br.ReadBytes((int)fs.Length),0,(int)fs.Length); 
br.Close(); 
fs.Close(); 

return doc; 

    4、在用戶端進行的工作。 
      首先引用此WebServices,例如命名為:WebSvs, 
string nVer = Start.GetService.GetVer();  
if(Application.ProductVersion.CompareTo(nVer)<=0) 
update(); 

在本代碼中 Start.GetService是WebSvs的一個Static 執行個體。首先檢查版本,將結果與目前的版本進行比較,如果為新版本則執行UpDate方法。 void update() 

this.statusBarPanel1.Text = "正在下載..."; 
System.Xml.XmlDocument doc = ((System.Xml.XmlDocument)Start.GetService.GetUpdateData()); 
doc.Save(Application.StartupPath + @"/update.xml"; 
System.Diagnostics.Process.Start(Application.StartupPath + @"/update.exe"; 
Close(); 
Application.Exit(); 
}這裡為了簡單起見,沒有使用非同步方法呼叫,當然使用非同步方法呼叫能更好的提高客戶體驗,這個需要讀者們自己去添加。:) update的作用是將升級的XML檔案下載下來,儲存為執行檔案目錄下的一個Update.xml檔案。任務完成,退出程式,等待Update.Exe 來進行升級。     5、Update.Exe 的內容。 private void Form1_Load(object sender, System.EventArgs e) 

System.Diagnostics.Process[] ps = System.Diagnostics.Process.GetProcesses(); 
foreach(System.Diagnostics.Process p in ps) 

//MessageBox.Show(p.ProcessName); 
if(p.ProcessName.ToLower()=="customerapplication" 

p.Kill(); 
break; 


XmlDocument doc = new XmlDocument(); 
doc.Load(Application.StartupPath + @"/update.xml"; 
XmlElement root = doc.DocumentElement; 
XmlNode updateNode = root.SelectSingleNode("filelist"; 
string path = updateNode.Attributes["sourcepath"].Value; 
int count = int.Parse(updateNode.Attributes["count"].Value); 
for(int i=0;i<count;i++) 

XmlNode itemNode = updateNode.ChildNodes[i]; 
string fileName = itemNode.Attributes["name"].Value; 
FileInfo fi = new FileInfo(fileName); 
fi.Delete(); 
//File.Delete(Application.StartupPath + @"/" + fileName); 
this.label1.Text = "正在更新: " + fileName + " (" + itemNode.Attributes["size"].Value + " ..."; 
FileStream fs = File.Open(fileName,FileMode.Create,FileAccess.Write); 
fs.Write(System.Convert.FromBase64String(itemNode.SelectSingleNode("value".InnerText),0,int.Parse(itemNode.Attributes["size"].Value)); 
fs.Close(); 

label1.Text = "更新完成"; 
File.Delete(Application.StartupPath + @"/update.xml"; 
label1.Text = "正在重新啟動應用程式..."; 
System.Diagnostics.Process.Start("CustomerApplication.exe"; 
Close(); 
Application.Exit(); 
} 這個代碼也很容易懂,首先就是找到主進程,如果沒有關閉,則用Process.Kill()來關閉主程式。然後則用一個XmlDocument來Load程式產生的update.xml檔案。用xml檔案裡指定的路徑和檔案名稱來產生指定的檔案,在這之前先前已經存在的檔案刪除。更新完畢後,則重新啟動主應用程式。這樣更新就完成了。  

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.