最近在重写了一个老的产品条码扫描程序,客户端数越有30个,因为经常有更新,c/s维护非常不方便,所以做一个自动更新程序特别有必要。
在网上随便找了找自动更新的方案,大多使用vs的发布/更新功能,不太喜欢此方案,觉得太麻烦,然后自己初步琢磨了一下如下方案。
现在比较喜欢的编程方式,先画图。
软件更新流程
安装包的制作
为了简单,采用了自解压压缩包的方式制作更新包,将整个新程序包打包为一个自解压压缩包。
以下是我的自解压包的参数设置,怎样制作自解压包网上有很多教程,可以自己去搜索。
常规-解压路径:"c:\barcodesys"
设置-解压后运行:"c:\barcodesys\updatelog.txt"
模式:隐藏启动对话框。
高级:添加快捷方式到桌面
源文件:"c:\barcodesys\barcodemain.exe"
快捷方式名:条码扫描程序
更新:解压替换、覆盖所有。
制作更新程序
因为软件在更新时update程序是一直在运行的,这样在自解压包解压时就会无法替换更新程序,所以将update程序作为一个单独的窗体程序来做,由主程序启动时调用。
制作更新程序前,首先解决怎样判断是否存在最新版本的问题,我采用ftp服务器上新建一个文本文档名称为svrver.txt,内容为最新版本号,本地程序文件夹中建立一个文本文档名称为version,内容为本地程序版本号。方法检查服务器版本号是否与本地版本一致,如果一致则表示不需要更新,否则相反,如下图
以下为更新程序下载文件部分的代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
|
long filesize = helper.getfilesize(dir+ "/release.exe" ); //获得更新程序文件大小 filesizes = filesize; if (filesize > 0) { updatelabeldelegate( "发现最新版本" + server + "!文件大小:" + filesize+ "字节" ); //更新界面label控件的委托 int readsize = 0; bool result = false ; using (filestream fs = new filestream( "release.exe" , filemode.create)) { string url = "ftp://" + ftpaddress + "/" +dir+ "/release.exe" ; console.writeline(url); ftpwebrequest request = (ftpwebrequest)ftpwebrequest.create( new uri(url)); request.credentials = new networkcredential(ftpusername, ftppassword); request.method = webrequestmethods.ftp.downloadfile; ; console.writeline(request); request.contentoffset = fs.length; updatelabeldelegate( "正在下载新版本... 0字节/" + filesize + "字节" ); using (ftpwebresponse response = (ftpwebresponse)request.getresponse()) { fs.position = fs.length; byte [] buffer = new byte [1024*2]; int count = response.getresponsestream().read(buffer, 0, buffer.length); while (count > 0) { fs.write(buffer, 0, count); count = response.getresponsestream().read(buffer, 0, buffer.length); readsize += count; double pa = ((( double )readsize / ( double )filesizes) * 100); updateprogdelegate(( int )pa) ; //更新进度条的委托 updateuilabel( "正在下载更新..." + (readsize) + "字节/" + (filesize) + "字节" ); application.doevents(); } updateuilabel( "下载完成,正在安装新版本,请稍等..." + (filesize) + "字节/" + (filesize) + "字节" ); updateprogdelegate(100); response.getresponsestream().close(); } } process p = process.start( "release.exe" ); //下载完成后调用自解压包 p.waitforexit(); //等待自解压包解压完成 p.startinfo.filename = "barcodemain.exe" ; //启动主程序 p.startinfo.arguments = "update" ; //传递主启动程序的参数,以免启动主程序后再次启动更新程序 p.start(); //启动 this .close(); //关闭更新程序窗口 |
主程序的部分小修改
因为每次启动时主程序时会调用update程序,同时关闭自身进程以准备更新,所以需要在主程序里做一些小事情,否则陷入启动主程序-启动更新程序-启动主程序的就无限循环中了。
1、修改主程序入口main方法,增加参数
1
2
3
4
5
6
7
|
[stathread] static void main( string [] arg) //接受参数 { application.enablevisualstyles(); application.setcompatibletextrenderingdefault( false ); application.run( new index(arg)); //传递给主程序窗体的构造函数 } |
2、主程序增加带参数的构造函数
1
2
3
4
5
6
7
8
9
10
|
private string [] targ ; //接收的参数 public index() { initializecomponent(); } public index( string [] arg) //带参数的构造函数 { initializecomponent(); targ = arg; } |
3、判断是否是否带参数启动
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
//检查执行参数 private static system.diagnostics.process p; if (targ.length==0) //如果没有传递参数,则调用update程序 { if (p == null ) { p = new system.diagnostics.process(); p.startinfo.filename = "updateprg.exe" ; p.start(); application.exit(); } else { if (p.hasexited) //是否正在运行 { p.start(); } } p.startinfo.windowstyle = system.diagnostics.processwindowstyle.normal; } |
最终的运行效果
最后推荐一个绿色版比较好用的简易ftp软件quick easy ftp server,足以满足更新所用。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:https://www.cnblogs.com/vice/p/8659817.html