DHT抓取程序开源地址:https://github.com/h31h31/H31DHTDEMO

数据处理程序开源地址:https://github.com/h31h31/H31DHTMgr

国外测试服务器: http://www.sosobta.com  大家可以给提点意见...

--------------------------------------------------------------------------------------------------------------------

关于自动更新,在.NET下面已经是很普通的事情,无非就是在服务器端保存配置好要更新的程序,然后客户端再写一个小程序来检测,有更新的则复制过来。

但现在问题是就一个程序,如何让程序自己进行更新而不用调用另外的程序,对于用户来说体验更好.

如何让程序自己更新自己的方法1:

  1. 1.首先程序exe自己下载覆盖自己肯定是不行的,会报“当前程序正在被另一个进程所使用”的系统错误;
  2.  
  3. 2.在进程中的程序不能覆盖自己,但是可以重命名,你从远程下载一个新的exe文件,重命名为xxx.exe.tmp;
  4.  
  5. 3.待下载完毕后,把旧的exe重命名一下,比如xxx.exe.old,然后把新的xxx.exe.tmp重命名为xxx.exe;
  6.  
  7. 4.重命名的方法:System.IO.File.Move(filepath + ".tmp", filepath);
  8.  
  9. 5.然后重启程序Application.Restart();在form_Load事件里面判断一下
  10. if (System.IO.File.Exists(AppDomain.CurrentDomain.BaseDirectory + @"xxx.exe.old"))
  11. System.IO.File.Delete(AppDomain.CurrentDomain.BaseDirectory + @"xxx.exe.old");
  12. 这样就可以了,如果做的再好点的话,可以把xxx.exe.old的文件属性设置为隐藏就好了.

如何让程序自己更新自己的方法2:

网上搜索的方法:

如果建一个bat文件,执行的是复制功能,然后再用一个bat来调用他,并且用一个bat文件去杀掉某一个文件,然后再复制新的,再启动是没有什么问题的吧。

添加了一个方法KillSelfThenRun()用于删除正在运行的主Exe,然后再重启新的主Exe。代码全部粘贴如下:

KillSelfThenRun
private void KillSelfThenRun()
{
string strXCopyFiles = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "XCopyFiles.bat");
using (StreamWriter swXcopy = File.CreateText(strXCopyFiles))
{
string strOriginalPath = tempUpdatePath.Substring(0, tempUpdatePath.Length - 1);
swXcopy.WriteLine(string.Format(@"
@echo off
xcopy /y/s/e/v " + strOriginalPath + " " + Directory.GetCurrentDirectory() +"", AppDomain.CurrentDomain.FriendlyName));
}
string filename = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "killmyself.bat");
using (StreamWriter bat = File.CreateText(filename))
{
// 自删除,自啟動
bat.WriteLine(string.Format(@"
@echo off
:selfkill
attrib -a -r -s -h ""{0}""
del ""{0}""
if exist ""{0}"" goto selfkill
call XCopyFiles.bat
del XCopyFiles.bat
del /f/q " + tempUpdatePath+Environment.NewLine + " rd " + tempUpdatePath +Environment.NewLine + " start " + mainAppExe +Environment.NewLine + " del %0 ", AppDomain.CurrentDomain.FriendlyName));
}

// 启动自删除批处理文件
ProcessStartInfo info = new ProcessStartInfo(filename);
info.WindowStyle = ProcessWindowStyle.Hidden;
Process.Start(info);

// 强制关闭当前进程
Environment.Exit(0);
}

第三步调用时,原程序本身就有一个Try的做法,就在Catch里面判断一下,如果出现IOException,就调用这个方法。

点击完成复制更新文件到应用程序目录
private void btnFinish_Click(object sender, System.EventArgs e)
{
this.Dispose();
//KillSelfThenRun();
try
{
CopyFile(tempUpdatePath, Directory.GetCurrentDirectory());
System.IO.Directory.Delete(tempUpdatePath, true);
}
catch (Exception ex)
{
if (ex.GetType() == typeof(IOException))
{
KillSelfThenRun();
}
else
{
MessageBox.Show(ex.Message.ToString());
}
}
if (true == this.isRun) Process.Start(mainAppExe);
}

经过分析后还是觉得第一种方法来得快些,简单易用:

先看使用方法:

在主程序界面里面增加一个定时器,然后用来延时来判断服务器上的程序是否有更新,如果在程序一开始就判断,程序会加载很慢,从而给用户体验不好.

  1. string m_localPath = AppDomain.CurrentDomain.BaseDirectory;
  2. string thisexename = Application.ExecutablePath.Replace(m_localPath, "");
  3. H31Updater update1 = new H31Updater();
  4. bool isupdate = update1.CheckServerUpdate(m_localPath, thisexename);
  5. if (isupdate)
  6. {
  7. DialogResult dlg = MessageBox.Show(this, "检测到服务器上有新版本,需要更新么?", "信息提示", MessageBoxButtons.OKCancel, MessageBoxIcon.Question);
  8. if (dlg == DialogResult.OK)
  9. {
  10. update1.Show();
  11. update1.UpdaterStart();
  12. }
  13. }

然后程序就会显示更新进度条:

程序更新后将自己命令为*.old,然后使用MOVE命令将新的更新为自己.

服务器上的返回XML文件列表方式:

  1. <?xml version="1.0" encoding="utf-8" ?>
  2. <AutoUpdater>
  3.  
  4. <UpdateInfo id="">
  5. <Version value = "1.0.0.3"/>
  6. </UpdateInfo>
  7. <!--升级文件列表-->
  8. <UpdateFileList>
  9. <UpdateFile>software\H31Thunder.exe</UpdateFile>
  1. <!--可以复制多行,更新自己目录下面的程序-->
  1.   <UpdateFile>software\H31Thunder.exe</UpdateFile>
  1. </UpdateFileList> </AutoUpdater>

上面配置的XML文件列表可以复制多行来执行多个更新.

现将自己经过测试的代码提供给大家讨论学习下,希望大家能够用到:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.ComponentModel;
  4. using System.Data;
  5. using System.Drawing;
  6. using System.Text;
  7. using System.Windows.Forms;
  8.  
  9. using System.Globalization;
  10. using System.IO;
  11. using System.Net;
  12. using System.Xml;
  13. using System.Threading;
  14.  
  15. namespace H31Thunder
  16. {
  17. public partial class H31Updater : Form
  18. {
  19. private WebClient downWebClient = new WebClient();
  20. private static string m_exePath = "";
  21. private static string m_exeName = "";
  22. private static string m_url = "http://h31bt.com/";
  23. private static string[] fileNames;
  24. private static int m_downNowPos;//已更新文件数
  25. private static string fileName;//当前文件名
  26. public H31Updater()
  27. {
  28. InitializeComponent();
  29. }
  30.  
  31. public bool CheckServerUpdate(string exePath,string exeName)
  32. {
  33. m_exePath = exePath;
  34. m_exeName = exeName;
  35. if (File.Exists(m_exePath + m_exeName + ".old"))
  36. File.Delete(m_exePath + m_exeName + ".old");
  37. string currentVersion = System.Reflection.Assembly.GetExecutingAssembly().GetName().Version.ToString();
  38. string theNewVersion = GetTheNewVersion(m_url + "version.asp?id=1");
  39. if (!String.IsNullOrEmpty(theNewVersion) && !String.IsNullOrEmpty(currentVersion))
  40. {
  41. if (string.Compare(theNewVersion,currentVersion)> )
  42. {
  43. return true;
  44. }
  45. }
  46. return false;
  47. }
  48.  
  49. /// <summary>
  50. /// 开始更新
  51. /// </summary>
  52. public void UpdaterStart()
  53. {
  54. try
  55. {
  56. float tempf;
  57. //委托下载数据时事件
  58. this.downWebClient.DownloadProgressChanged += delegate(object wcsender, DownloadProgressChangedEventArgs ex)
  59. {
  60. this.label2.Text = String.Format(CultureInfo.InvariantCulture,"正在下载:{0} [ {1}/{2} ]",fileName,ConvertSize(ex.BytesReceived),ConvertSize(ex.TotalBytesToReceive));
  61.  
  62. tempf = ((float)(m_downNowPos) / fileNames.Length);
  63. this.progressBar2.Value = ex.ProgressPercentage;
  64. this.progressBar1.Value = Convert.ToInt32(tempf * );
  65. };
  66. //委托下载完成时事件
  67. this.downWebClient.DownloadFileCompleted += delegate(object wcsender, AsyncCompletedEventArgs ex)
  68. {
  69. if (ex.Error != null)
  70. {
  71. MeBox(ex.Error.Message);
  72. }
  73. else
  74. {
  75. if (fileNames.Length > m_downNowPos)
  76. {
  77. DownloadFile(m_downNowPos);
  78. }
  79. else
  80. {
  81. this.progressBar1.Value = this.progressBar1.Maximum;
  82. this.progressBar2.Value = this.progressBar2.Maximum;
  83. Thread.Sleep();
  84. UpdaterClose();
  85. }
  86. }
  87. };
  88.  
  89. m_downNowPos = ;
  90. if (fileNames != null)
  91. DownloadFile();
  92. }
  93. catch (WebException ex)
  94. {
  95. MeBox(ex.Message);
  96. }
  97. }
  98.  
  99. /// <summary>
  100. /// 下载文件
  101. /// </summary>
  102. /// <param name="arry">下载序号</param>
  103. private void DownloadFile(int arry)
  104. {
  105. try
  106. {
  107. m_downNowPos++;
  108. int findpos = fileNames[arry].LastIndexOf("/");
  109. if (findpos != -)
  110. fileName = fileNames[arry].Substring(findpos+, fileNames[arry].Length - findpos-);
  111. else
  112. fileName = fileNames[arry];
  113. this.label1.Text = String.Format(CultureInfo.InvariantCulture, "更新进度 {0}/{1}", m_downNowPos, fileNames.Length);
  114.  
  115. this.progressBar2.Value = ;
  116. string weburl = m_url + fileNames[arry];
  117. string savename=m_exePath + fileName + ".new";
  118. this.downWebClient.DownloadFileAsync(new Uri(weburl), savename);
  119. }
  120. catch (Exception ex)
  121. {
  122. MeBox(ex.Message);
  123. }
  124. }
  125.  
  126. /// <summary>
  127. /// 关闭程序
  128. /// </summary>
  129. private static void UpdaterClose()
  130. {
  131. try
  132. {
  133. if (File.Exists(m_exePath + m_exeName + ".old"))
  134. File.Delete(m_exePath + m_exeName + ".old");
  135. for(int i=;i<fileNames.Length;i++)
  136. {
  137. int findpos = fileNames[i].LastIndexOf("/");
  138. string savename = fileNames[i];
  139. if(findpos!=-)
  140. savename = fileNames[i].Substring(findpos+, fileNames[i].Length - findpos-);
  141. if (File.Exists(m_exePath + savename + ".old"))
  142. File.Delete(m_exePath + savename + ".old");
  143. File.Move(m_exePath + savename, m_exePath + savename + ".old");
  144. File.Move(m_exePath + savename + ".new", m_exePath + savename);
  145. File.SetAttributes(m_exePath + savename + ".old", FileAttributes.Archive | FileAttributes.Hidden); 
  146. }
  147. }
  148. catch (Exception ex)
  149. {
  150. //if (ex.GetType() == typeof(IOException))
  151. //{
  152.  
  153. //}
  154. //else
  155. MeBox(ex.Message);
  156. }
  157.  
  158. Application.Exit();
  159. }
  160.  
  161. /// <summary>
  162. /// 判断软件的更新日期
  163. /// </summary>
  164. /// <param name="Dir">服务器地址</param>
  165. /// <returns>返回日期</returns>
  166. private static string GetTheNewVersion(string webUrl)
  167. {
  168. string NewVersion = "";
  169. try
  170. {
  171. WebClient wc = new WebClient();
  172. Stream getStream = wc.OpenRead(webUrl);
  173. StreamReader streamReader = new StreamReader(getStream, Encoding.UTF8);
  174. string xmltext = streamReader.ReadToEnd();
  175. streamReader.Close();
  176.  
  177. XmlDocument xmlDoc = new XmlDocument();
  178. xmlDoc.InnerXml = xmltext;
  179. XmlNode root = xmlDoc.SelectSingleNode("//Version");
  180. if (root.Name == "Version")
  181. {
  182. NewVersion = root.Attributes["value"].Value.ToString();
  183. }
  184. XmlNodeList root1 = xmlDoc.SelectNodes("//UpdateFileList");
  185. if (root1 != null)
  186. {
  187. fileNames = new string[root1.Count];
  188. int i=;
  189. foreach (XmlNode temp in root1)
  190. {
  191. fileNames[i] = temp.InnerText.Replace("\\","/");
  192. i++;
  193. }
  194. }
  195.  
  196. }
  197. catch (WebException ex)
  198. {
  199. //MeBox(ex.Message);
  200. }
  201. return NewVersion;
  202. }
  203.  
  204. /// <summary>
  205. /// 弹出提示框
  206. /// </summary>
  207. /// <param name="txt">输入提示信息</param>
  208. private static void MeBox(string txt)
  209. {
  210. MessageBox.Show(txt, "提示信息", MessageBoxButtons.OK, MessageBoxIcon.Asterisk, MessageBoxDefaultButton.Button1, MessageBoxOptions.DefaultDesktopOnly);
  211. }
  212. /// <summary>
  213. /// 转换字节大小
  214. /// </summary>
  215. /// <param name="byteSize">输入字节数</param>
  216. /// <returns>返回值</returns>
  217. private static string ConvertSize(long byteSize)
  218. {
  219. string str = "";
  220. float tempf = (float)byteSize;
  221. if (tempf / > )
  222. {
  223. if ((tempf / ) / > )
  224. {
  225. str = ((tempf / ) / ).ToString("##0.00", CultureInfo.InvariantCulture) + "MB";
  226. }
  227. else
  228. {
  229. str = (tempf / ).ToString("##0.00", CultureInfo.InvariantCulture) + "KB";
  230. }
  231. }
  232. else
  233. {
  234. str = tempf.ToString(CultureInfo.InvariantCulture) + "B";
  235. }
  236. return str;
  237. }
  238.  
  239. }
  240. }

程序自动更新类代码

如果大家有什么更好的方法,请指教.谢谢.

下载程序:H31Thunder.rar

下一篇给大家介绍上面图中显示如何从迅雷服务器上偷用视频图片地址的方法.不信的话大家可以去查看下图片的 http://www.sosobta.com 网址是不是自己服务器上的...

当然希望大家多多推荐哦...大家的推荐才是下一篇介绍的动力...

[搜片神器]winform程序自己如何更新自己的方法代码的更多相关文章

  1. MFC编辑框接收数据动态更新与刷新方法代码示例-如何让编辑框内容实时更新

    MFC编辑框接收数据动态更新与刷新方法代码示例-如何让编辑框内容实时更新 关键代码: //发送数据通知 //from txwtech@163.com LRESULT CCommSampleDlg::O ...

  2. 分析nuget源码,用nuget + nuget.server实现winform程序的自动更新

    源起 (个人理解)包管理最开始应该是从java平台下的maven开始吧,因为java的开发大多数是基于开源组件开发的,一个开源包在使用时很可能要去依赖其他的开源包,而且必须是特定的版本才可以.以往在找 ...

  3. [搜片神器]之DHT网络爬虫的C++程序初步开源

    回应大家的要求,特地整理了一开始自己整合的代码,这样最简单,最直接的可以分析流程,至于文章里面提供的程序界面更多,需要大家自己开发. 谢谢园子朋友的支持,已经找到个VPS进行测试,国外的服务器: ht ...

  4. [C#搜片神器] 之P2P中DHT网络爬虫原理

    继续接着上一篇写:使用C#实现DHT磁力搜索的BT种子后端管理程序+数据库设计(开源)[搜片神器] 昨天由于开源的时候没有注意运行环境,直接没有考虑下载BT种子文件时生成子文件夹,可能导致有的朋友运行 ...

  5. [搜片神器]之DHT网络爬虫的代码实现方法

    继续接着第一篇写:使用C#实现DHT磁力搜索的BT种子后端管理程序+数据库设计(开源)[搜片神器] 谢谢园子朋友的支持,已经找到个VPS进行测试,国外的服务器: http://www.sosobta. ...

  6. C# winform程序怎么打包成安装项目(VS2010图解)

    作为研发人员,在本机上开发的winform.wpf或者控制台程序需要发给其他人测试时候,一般需要对其进行打包生成setup安装文件,根据网上查找的资料并结合自己打包成功,记录如下: 注:本程序是一个利 ...

  7. 用winform程序来了解委托和事件

    一.浅谈委托 如果有个过winform 和webform 程序开发的小伙伴一定有个这样的感觉吧,点击Button直接就执行了那个方法,到此他是怎么实现了的呢,大家有考虑过没有? 回到正题,什么是委托呢 ...

  8. 黄聪:C#Winform程序如何发布并自动升级(图解)

    有不少朋友问到C#Winform程序怎么样配置升级,怎么样打包,怎么样发布的,在这里我解释一下打包和发布关于打包的大家可以看我的文章C# winform程序怎么打包成安装项目(图解)其实打包是打包,发 ...

  9. 【转】C#Winform程序如何发布并自动升级(图解)

    有不少朋友问到C#Winform程序怎么样配置升级,怎么样打包,怎么样发布的,在这里我解释一下打包和发布关于打包的大家可以看我的文章C# winform程序怎么打包成安装项目(图解)其实打包是打包,发 ...

随机推荐

  1. MVC 使用Jquery实现AJax

    View <script type="text/javascript"> function GetTime() { $.get("Home/GetTime&q ...

  2. swift 闭包简写实际参数名$0、$1等理解

    Swift 自动对行内闭包提供简写实际参数名,你也可以通过 $0 , $1 , $2 等名字来引用闭包的实际参数值. 如果你在闭包表达式中使用这些简写实际参数名,那么你可以在闭包的实际参数列表中忽略对 ...

  3. 20150301—ASP.NET的Repeater

    Repeater与GridView等数据列表一样,都是用来显示数据库的信息的,其中Repeater是最基本的列表形式,其用法也比较灵活. 一.Repeater的位置: 工具箱-数据-Repeater ...

  4. WCF之契约

    消息交换的双方,为了进行消息交换,而定义的一些数据交换规则,称之为契约. 契约只约束规则,不管实现. 契约对客户端和服务器的要求. 服务器:定义和实现契约.构建ServiceHost实例,然后暴露En ...

  5. (转)Linux概念架构的理解

    英文原文:Conceptual Architecture of the Linux Kernel 摘要 Linux kernel成功的两个原因:(1)架构设计支持大量的志愿开发者加入到开发过程中:(2 ...

  6. void指针

    指针有两个属性:指向变量/对象的地址 和长度 但是指针只存储地址,长度则取决于指针的类型 编译器根据指针的类型从指针指向的地址向后寻址 指针类型不同则寻址范围也不同,比如: int*从指定地址向后寻找 ...

  7. <Linux系统hostname命令详解>

    hostname命令的用法的小知识我们都知道hostname命令是查看主机名和修改主机名的. [root@apache ~]# hostname  //查看本机的主机名apache.example.c ...

  8. Android Studio添加jar包

    1.先把jar包复制到项目的lib下,

  9. seajs构建方法

    标准构建 如果项目遵循推荐的标准目录结构: foo-module/   |-- dist                    //存放构建好的文件   |-- src                 ...

  10. 如何设置SecureCRT通过代理连接SSH[转]

    http://blog.didu.me/article/84 公司限制了连接外网的端口和IP,只能通过proxy 连接.刚配置了一下 secureCRT 连接外网,貌似速度还是不错,写出来共享下. 如 ...