在客户端实现更新操作

涉及到三个技术:

1.xml文件的解析

2.HttpURLConnection连接

3.文件流I/O

这里创建一个解析xml文件的服务类:ParXmlService.java

  1. package com.xiaowu.news.update;
  2. import java.io.InputStream;
  3. import java.util.HashMap;
  4. import javax.xml.parsers.DocumentBuilder;
  5. import javax.xml.parsers.DocumentBuilderFactory;
  6. import org.w3c.dom.Document;
  7. import org.w3c.dom.Element;
  8. import org.w3c.dom.Node;
  9. import org.w3c.dom.NodeList;
  10. public class ParseXmlService {
  11. public HashMap<String, String> parseXml (InputStream inStream) throws Exception{
  12. HashMap<String, String> hashMap = new HashMap<String, String>();
  13. //创建DocumentBuilderFactory,该对象将创建DocumentBuilder。
  14. DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
  15. //创建DocumentBuilder,DocumentBuilder将实际进行解析以创建Document对象
  16. DocumentBuilder builder = factory.newDocumentBuilder();
  17. //解析该文件以创建Document对象
  18. Document document = builder.parse(inStream);
  19. //获取XML文件根节点
  20. Element root = document.getDocumentElement();
  21. //获得所有子节点
  22. NodeList childNodes = root.getChildNodes();
  23. for(int i = 0; i < childNodes.getLength(); i++) {
  24. Node childNode = (Node) childNodes.item(i);
  25. if(childNode.getNodeType() == Node.ELEMENT_NODE) {
  26. Element childElement = (Element) childNode;
  27. //版本号
  28. if("version".equals(childElement.getNodeName())) {
  29. hashMap.put("version", childElement.getFirstChild().getNodeValue());
  30. //软件名称
  31. } else if("name".equals(childElement.getNodeName())) {
  32. hashMap.put("name", childElement.getFirstChild().getNodeValue());
  33. //下载地址
  34. } else if("url".equals(childElement.getNodeName())) {
  35. hashMap.put("url", childElement.getFirstChild().getNodeValue());
  36. }
  37. }
  38. }
  39. return hashMap;
  40. }
  41. }

实现更新操作的管理类:UpdateManager.java

  1. package com.xiaowu.news.update;
  2. import java.io.File;
  3. import java.io.FileOutputStream;
  4. import java.io.IOException;
  5. import java.io.InputStream;
  6. import java.net.HttpURLConnection;
  7. import java.net.MalformedURLException;
  8. import java.net.URL;
  9. import java.util.HashMap;
  10. import javax.net.ssl.HttpsURLConnection;
  11. import android.app.AlertDialog;
  12. import android.app.AlertDialog.Builder;
  13. import android.app.Dialog;
  14. import android.content.Context;
  15. import android.content.DialogInterface;
  16. import android.content.Intent;
  17. import android.content.DialogInterface.OnClickListener;
  18. import android.content.pm.PackageManager.NameNotFoundException;
  19. import android.net.Uri;
  20. import android.os.Environment;
  21. import android.os.Handler;
  22. import android.view.LayoutInflater;
  23. import android.view.View;
  24. import android.widget.ProgressBar;
  25. import android.widget.Toast;
  26. import com.xiaowu.news.R;
  27. /**
  28. *
  29. * @author wwj
  30. * @date 2012/11/17
  31. * 实现软件更新的管理类
  32. */
  33. public class UpdateManager {
  34. //下载中...
  35. private static final int DOWNLOAD = 1;
  36. //下载完成
  37. private static final int DOWNLOAD_FINISH = 2;
  38. //保存解析的XML信息
  39. HashMap<String , String> mHashMap;
  40. //下载保存路径
  41. private String mSavePath;
  42. //记录进度条数量
  43. private int progress;
  44. //是否取消更新
  45. private boolean cancelUpdate = false;
  46. //上下文对象
  47. private Context mContext;
  48. //进度条
  49. private ProgressBar mProgressBar;
  50. //更新进度条的对话框
  51. private Dialog mDownloadDialog;
  52. private Handler mHandler = new Handler() {
  53. public void handleMessage(android.os.Message msg) {
  54. switch(msg.what){
  55. //下载中。。。
  56. case DOWNLOAD:
  57. //更新进度条
  58. System.out.println(progress);
  59. mProgressBar.setProgress(progress);
  60. break;
  61. //下载完成
  62. case DOWNLOAD_FINISH:
  63. // 安装文件
  64. installApk();
  65. break;
  66. }
  67. };
  68. };
  69. public UpdateManager(Context context) {
  70. super();
  71. this.mContext = context;
  72. }
  73. /**
  74. * 检测软件更新
  75. */
  76. public void checkUpdate() {
  77. if (isUpdate()) {
  78. //显示提示对话框
  79. showNoticeDialog();
  80. } else {
  81. Toast.makeText(mContext, R.string.soft_update_no, Toast.LENGTH_SHORT).show();
  82. }
  83. }
  84. private void showNoticeDialog() {
  85. // TODO Auto-generated method stub
  86. //构造对话框
  87. AlertDialog.Builder builder = new Builder(mContext);
  88. builder.setTitle(R.string.soft_update_title);
  89. builder.setMessage(R.string.soft_update_info);
  90. //更新
  91. builder.setPositiveButton(R.string.soft_update_updatebtn, new OnClickListener() {
  92. @Override
  93. public void onClick(DialogInterface dialog, int which) {
  94. // TODO Auto-generated method stub
  95. dialog.dismiss();
  96. // 显示下载对话框
  97. showDownloadDialog();
  98. }
  99. });
  100. // 稍后更新
  101. builder.setNegativeButton(R.string.soft_update_later, new OnClickListener() {
  102. @Override
  103. public void onClick(DialogInterface dialog, int which) {
  104. // TODO Auto-generated method stub
  105. dialog.dismiss();
  106. }
  107. });
  108. Dialog noticeDialog = builder.create();
  109. noticeDialog.show();
  110. }
  111. private void showDownloadDialog() {
  112. // 构造软件下载对话框
  113. AlertDialog.Builder builder = new Builder(mContext);
  114. builder.setTitle(R.string.soft_updating);
  115. // 给下载对话框增加进度条
  116. final LayoutInflater inflater = LayoutInflater.from(mContext);
  117. View view = inflater.inflate(R.layout.softupdate_progress, null);
  118. mProgressBar = (ProgressBar) view.findViewById(R.id.update_progress);
  119. builder.setView(view);
  120. builder.setNegativeButton(R.string.soft_update_cancel, new OnClickListener() {
  121. @Override
  122. public void onClick(DialogInterface dialog, int which) {
  123. // TODO Auto-generated method stub
  124. dialog.dismiss();
  125. // 设置取消状态
  126. cancelUpdate = true;
  127. }
  128. });
  129. mDownloadDialog = builder.create();
  130. mDownloadDialog.show();
  131. //下载文件
  132. downloadApk();
  133. }
  134. /**
  135. * 下载APK文件
  136. */
  137. private void downloadApk() {
  138. // TODO Auto-generated method stub
  139. // 启动新线程下载软件
  140. new DownloadApkThread().start();
  141. }
  142. /**
  143. * 检查软件是否有更新版本
  144. * @return
  145. */
  146. public boolean isUpdate() {
  147. // 获取当前软件版本
  148. int versionCode = getVersionCode(mContext);
  149. //把version.xml放到网络上,然后获取文件信息
  150. InputStream inStream = ParseXmlService.class.getClassLoader().getResourceAsStream("version.xml");
  151. // 解析XML文件。 由于XML文件比较小,因此使用DOM方式进行解析
  152. ParseXmlService service = new ParseXmlService();
  153. try {
  154. mHashMap = service.parseXml(inStream);
  155. } catch (Exception e) {
  156. // TODO: handle exception
  157. e.printStackTrace();
  158. }
  159. if(null != mHashMap) {
  160. int serviceCode = Integer.valueOf(mHashMap.get("version"));
  161. //版本判断
  162. if(serviceCode > versionCode) {
  163. return true;
  164. }
  165. }
  166. return false;
  167. }
  168. /**
  169. * 获取软件版本号
  170. * @param context
  171. * @return
  172. */
  173. private int getVersionCode(Context context) {
  174. // TODO Auto-generated method stub
  175. int versionCode = 0;
  176. // 获取软件版本号,对应AndroidManifest.xml下android:versionCode
  177. try {
  178. versionCode = context.getPackageManager().getPackageInfo(
  179. "com.xiaowu.news", 0).versionCode;
  180. } catch (NameNotFoundException e) {
  181. // TODO Auto-generated catch block
  182. e.printStackTrace();
  183. }
  184. return versionCode;
  185. }
  186. /**
  187. * 下载文件线程
  188. * @author Administrator
  189. *
  190. */
  191. private class DownloadApkThread extends Thread {
  192. @Override
  193. public void run() {
  194. try
  195. {
  196. //判断SD卡是否存在,并且是否具有读写权限
  197. if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED))
  198. {
  199. // 获取SDCard的路径
  200. String sdpath = Environment.getExternalStorageDirectory() + "/";
  201. mSavePath = sdpath + "download";
  202. URL url = new URL(mHashMap.get("url"));
  203. // 创建连接
  204. HttpURLConnection conn = (HttpURLConnection) url.openConnection();
  205. conn.connect();
  206. // 获取文件大小
  207. int length = conn.getContentLength();
  208. // 创建输入流
  209. InputStream is = conn.getInputStream();
  210. File file = new File(mSavePath);
  211. // 如果文件不存在,新建目录
  212. if (!file.exists())
  213. {
  214. file.mkdir();
  215. }
  216. File apkFile = new File(mSavePath, mHashMap.get("name"));
  217. FileOutputStream fos = new FileOutputStream(apkFile);
  218. int count = 0;
  219. // 缓存
  220. byte buf[] = new byte[1024];
  221. // 写入到文件中
  222. do
  223. {
  224. int numread = is.read(buf);
  225. count += numread;
  226. // 计算进度条的位置
  227. progress = (int) (((float) count / length) * 100);
  228. // 更新进度
  229. mHandler.sendEmptyMessage(DOWNLOAD);
  230. if (numread <= 0)
  231. {
  232. // 下载完成
  233. mHandler.sendEmptyMessage(DOWNLOAD_FINISH);
  234. break;
  235. }
  236. // 写入文件
  237. fos.write(buf, 0, numread);
  238. } while (!cancelUpdate);//点击取消就停止下载
  239. fos.close();
  240. is.close();
  241. }
  242. } catch (MalformedURLException e)
  243. {
  244. e.printStackTrace();
  245. } catch (IOException e)
  246. {
  247. e.printStackTrace();
  248. }
  249. // 取消下载对话框显示
  250. mDownloadDialog.dismiss();
  251. }
  252. }
  253. /**
  254. * 安装APK文件
  255. */
  256. private void installApk()
  257. {
  258. File apkfile = new File(mSavePath, mHashMap.get("name"));
  259. if (!apkfile.exists())
  260. {
  261. return;
  262. }
  263. Intent i = new Intent(Intent.ACTION_VIEW);
  264. i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive");
  265. mContext.startActivity(i);
  266. }
  267. }

原文地址:http://blog.csdn.net/wwj_748/article/details/8195565

Android - 软件自动更新的实现(转)的更多相关文章

  1. Android - 软件自动更新的实现

    转自:http://blog.csdn.net/wwj_748/article/details/8195565 接触到一个很实用的技术,那就是软件自动更新.一般开发者是通过自行在应用平台添加更新版本的 ...

  2. Android 软件自动更新功能实现的方法

    相信所有的用户都遇到过软件提醒更新的情况,下面就将实现此功能 首先看一下程序目录结构 步骤: 1.新建一个类UpdateManger,用于显示提示更新 详细出处参考:http://www.jb51.n ...

  3. Android实现自动更新功能

    Android实现自动更新功能 Android自动更新的功能可以使用第三方的SDK来实现,但是类似友盟,就不支持x86手机的自动更新,科大讯飞,弹窗是全局的,小米手机就会默认把弹窗权限关掉不允许弹出提 ...

  4. Android应用自动更新功能的实现!

    Android应用自动更新功能的实现!http://blog.csdn.net/android_tutor/article/details/7015986 private static final i ...

  5. Android应用自动更新功能的代码实现

    由于Android项目开源所致,市面上出现了N多安卓软件市场.为了让我们开发的软件有更多的用户使用,我们需要向N多市场发布,软件升级后,我们也必须到安卓市场上进行更新,给我们增加了工作量.因此我们有必 ...

  6. 使用七牛云存储实现Android的自动更新

    为了修复Bug或更新软件,我们通常需要实现自动更新,没有哪一个牛逼的人能够搞到每一个用户的机子去帮他们更新. 1.自动更新的流程 我们将了解一下自动更新的思路.既然软件要自动更新,那么它必须知道自己是 ...

  7. Android 应用自动更新功能的代码

    由于Android项目开源所致,市面上出现了N多安卓软件市场.为了让我们开发的软件有更多的用户使用,我们需要向N多市场发布,软件升级后,我们也必须到安卓市场上进行更新,给我们增加了工作量.因此我们有必 ...

  8. Android应用自动更新功能的实现!!!

    自动更新功能的实现原理,就是我们事先和后台协商好一个接口,我们在应用的主Activity里,去访问这个接口,如果需要更新,后台会返回一些数据(比如,提示语:最新版本的url等).然后我们给出提示框,用 ...

  9. Android应用自动更新功能的代码实现(转)

    由于Android项目开源所致,市面上出现了N多安卓软件市场.为了让我们开发的软件有更多的用户使用,我们需要向N多市场发布,软件升级后,我们也必须到安卓市场上进行更新,给我们增加了工作量.因此我们有必 ...

随机推荐

  1. ThinkPHP接入支付宝支付功能

    最近做系统,需要实现在线支付功能,毫不犹豫,选择的是支付宝的接口支付功能.这里我用的是即时到帐的接口,具体实现的步骤如下: 一.下载支付宝接口包 下载地址:https://b.alipay.com/o ...

  2. 根据select不同的选项实现相应input框添加项的显示

    实现效果: @1.单击包时,显示包时的添加项 @2.单击包里程,显示包里程的添加项 二  代码实现: 给select添加change事件 获取当前select的value 根据value判断对象显示其 ...

  3. ACM竞赛常用STL(一)

    全排列函数next_permutation STL 中专门用于排列的函数(可以处理存在重复数据集的排列问题) 头文件:#include <algorithm> using namespac ...

  4. hdu 1229 超级大水题

      Time Limit: 1000MS   Memory Limit: 32768KB   64bit IO Format: %I64d & %I64u Submit Status Desc ...

  5. EntityFramework Core使用PostgreSQL

    EntityFramework Core使用PostgreSQL 0 Asp.Net Core 项目实战之权限管理系统(0) 无中生有 1 Asp.Net Core 项目实战之权限管理系统(1) 使用 ...

  6. Python模块解析之SocketServer(三)——模块思想

            SocketServer 体系 由两个部分构成 BaseServer 和 BaseRequestHandler.思想很简单 BaseServer接受请求,将请求交给BaseReques ...

  7. EasyPR--开发详解

    我正在做一个开源的中文车牌识别系统,Git地址为:https://github.com/liuruoze/EasyPR. 我给它取的名字为EasyPR,也就是Easy to do Plate Reco ...

  8. [BZOJ 1801] [Ahoi2009]chess 中国象棋 【DP】

    题目链接:BZOJ - 1801 题目分析 对于50%的数据是可以直接状压 DP 的. 对于100%的数据,使用递推的 DP .(或者这只叫递推不叫 DP ?) 可以发现,每一行和每一列的棋子个数不能 ...

  9. table 的边框变细

    table { border-width: 0px 0px 1px 1px; border-style:solid;border-color:black} td { border-width: 1px ...

  10. Volatile vs. Interlocked vs. lock

    今天在stackoverflow上看到一个关于Volatile, Interlock, Lock的问题,发现回答的特别好,所以就想到把它翻译一下, 希望给那些对它们有疑惑的人提供点帮助 :假设有一个类 ...