热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android软件自动更新实现代码

这篇文章主要为大家详细介绍了Android软件自动更新实现代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

如何实现软件自动更新,下面是具体实例:

效果图: 

Android应用自动更新的实现   

Android应用自动更新的实现

Android应用自动更新的实现    

具体步骤:

1、在服务器上部署更新所用的xml文件:version.xml


 2
 baiduxinwen.apk
 http://gdown.baidu.com/data/wisegame/e5f5c3b8e59401c8/baiduxinwen.apk

2、在客户端实现更新操作

涉及到三个技术:

   1.xml文件的解析

   2.HttpURLConnection连接

   3.文件流I/O

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

package com.xiaowu.news.update;

import java.io.InputStream;
import java.util.HashMap;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;

import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

public class ParseXmlService {
 public HashMap parseXml (InputStream inStream) throws Exception{
 HashMap hashMap = new HashMap();
 //创建DocumentBuilderFactory,该对象将创建DocumentBuilder。
 DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
 //创建DocumentBuilder,DocumentBuilder将实际进行解析以创建Document对象
 DocumentBuilder builder = factory.newDocumentBuilder();
 //解析该文件以创建Document对象
 Document document = builder.parse(inStream);
 //获取XML文件根节点 
 Element root = document.getDocumentElement();
 //获得所有子节点
 NodeList childNodes = root.getChildNodes();
 for(int i = 0; i 


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

package com.xiaowu.news.update;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.HashMap;

import javax.net.ssl.HttpsURLConnection;

import android.app.AlertDialog;
import android.app.AlertDialog.Builder;
import android.app.Dialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.DialogInterface.OnClickListener;
import android.content.pm.PackageManager.NameNotFoundException;
import android.net.Uri;
import android.os.Environment;
import android.os.Handler;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.ProgressBar;
import android.widget.Toast;

import com.xiaowu.news.R;

/**
 * 
 * @author wwj
 * @date 2012/11/17
 * 实现软件更新的管理类
 */
public class UpdateManager {
 
 //下载中...
 private static final int DOWNLOAD = 1;
 //下载完成
 private static final int DOWNLOAD_FINISH = 2;
 //保存解析的XML信息
 HashMap mHashMap;
 //下载保存路径
 private String mSavePath;
 //记录进度条数量
 private int progress;
 //是否取消更新
 private boolean cancelUpdate = false;
 //上下文对象
 private Context mContext;
 //进度条
 private ProgressBar mProgressBar;
 //更新进度条的对话框
 private Dialog mDownloadDialog;
 
 
 private Handler mHandler = new Handler() {
 public void handleMessage(android.os.Message msg) {
  switch(msg.what){
  //下载中。。。
  case DOWNLOAD:
  //更新进度条
  System.out.println(progress);
  mProgressBar.setProgress(progress);
  break;
  //下载完成
  case DOWNLOAD_FINISH:
  // 安装文件
  installApk();
  break;
  }
 };
 };


 public UpdateManager(Context context) {
 super();
 this.mCOntext= context;
 }
 
 
 /**
 * 检测软件更新
 */
 public void checkUpdate() {
 if (isUpdate()) {
  //显示提示对话框
  showNoticeDialog();
 } else {
  Toast.makeText(mContext, R.string.soft_update_no, Toast.LENGTH_SHORT).show();
 }
 
 }
 
 private void showNoticeDialog() {
 // TODO Auto-generated method stub
 //构造对话框
 AlertDialog.Builder builder = new Builder(mContext);
 builder.setTitle(R.string.soft_update_title);
 builder.setMessage(R.string.soft_update_info);
 //更新
 builder.setPositiveButton(R.string.soft_update_updatebtn, new OnClickListener() {
  
  @Override
  public void onClick(DialogInterface dialog, int which) {
  // TODO Auto-generated method stub
  dialog.dismiss();
  // 显示下载对话框
  showDownloadDialog();
  }
 });
 // 稍后更新
 builder.setNegativeButton(R.string.soft_update_later, new OnClickListener() {
  
  @Override
  public void onClick(DialogInterface dialog, int which) {
  // TODO Auto-generated method stub
  dialog.dismiss();
  }
 });
 Dialog noticeDialog = builder.create();
 noticeDialog.show();
 }
 
 private void showDownloadDialog() {
 // 构造软件下载对话框
 AlertDialog.Builder builder = new Builder(mContext);
 builder.setTitle(R.string.soft_updating);
 // 给下载对话框增加进度条
 final LayoutInflater inflater = LayoutInflater.from(mContext);
 View view = inflater.inflate(R.layout.softupdate_progress, null);
 mProgressBar = (ProgressBar) view.findViewById(R.id.update_progress);
 builder.setView(view);
 builder.setNegativeButton(R.string.soft_update_cancel, new OnClickListener() {
  
  @Override
  public void onClick(DialogInterface dialog, int which) {
  // TODO Auto-generated method stub
  dialog.dismiss();
  // 设置取消状态
  cancelUpdate = true;
  }
 });
 mDownloadDialog = builder.create();
 mDownloadDialog.show();
 //下载文件
 downloadApk();
 }
 
 /**
 * 下载APK文件
 */
 private void downloadApk() {
 // TODO Auto-generated method stub
 // 启动新线程下载软件
 new DownloadApkThread().start();
 }


 /**
 * 检查软件是否有更新版本
 * @return
 */
 public boolean isUpdate() {
 // 获取当前软件版本
 int versiOnCode= getVersionCode(mContext);
 //把version.xml放到网络上,然后获取文件信息
 InputStream inStream = ParseXmlService.class.getClassLoader().getResourceAsStream("version.xml");
 // 解析XML文件。 由于XML文件比较小,因此使用DOM方式进行解析
 ParseXmlService service = new ParseXmlService();
 try {
  mHashMap = service.parseXml(inStream);
 } catch (Exception e) {
  // TODO: handle exception
  e.printStackTrace();
 }
 if(null != mHashMap) {
  int serviceCode = Integer.valueOf(mHashMap.get("version"));
  //版本判断
  if(serviceCode > versionCode) {
  return true;
  }
 }
 return false;
 }

 /**
 * 获取软件版本号
 * @param context
 * @return
 */
 private int getVersionCode(Context context) {
 // TODO Auto-generated method stub
 int versiOnCode= 0;

 // 获取软件版本号,对应AndroidManifest.xml下android:versionCode
 try {
  versiOnCode= context.getPackageManager().getPackageInfo(
   "com.xiaowu.news", 0).versionCode;
 } catch (NameNotFoundException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
 }
 return versionCode;
 }
 
 /**
 * 下载文件线程
 * @author Administrator
 *
 */
 private class DownloadApkThread extends Thread {
 @Override
 public void run() {
  try
  {
  //判断SD卡是否存在,并且是否具有读写权限
  if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED))
  {
   // 获取SDCard的路径
   String sdpath = Environment.getExternalStorageDirectory() + "/";
   mSavePath = sdpath + "download";
   URL url = new URL(mHashMap.get("url"));
   // 创建连接
   HttpURLConnection cOnn= (HttpURLConnection) url.openConnection();
   conn.connect();
   // 获取文件大小
   int length = conn.getContentLength();
   // 创建输入流
   InputStream is = conn.getInputStream();

   File file = new File(mSavePath);
   // 如果文件不存在,新建目录
   if (!file.exists())
   {
   file.mkdir();
   }
   File apkFile = new File(mSavePath, mHashMap.get("name"));
   FileOutputStream fos = new FileOutputStream(apkFile);
   int count = 0;
   // 缓存
   byte buf[] = new byte[1024];
   // 写入到文件中
   do
   {
   int numread = is.read(buf);
   count += numread;
   // 计算进度条的位置
   progress = (int) (((float) count / length) * 100);
   // 更新进度
   mHandler.sendEmptyMessage(DOWNLOAD);
   if (numread <= 0)
   {
    // 下载完成
    mHandler.sendEmptyMessage(DOWNLOAD_FINISH);
    break;
   }
   // 写入文件
   fos.write(buf, 0, numread);
   } while (!cancelUpdate);//点击取消就停止下载
   fos.close();
   is.close();
  }
  } catch (MalformedURLException e)
  {
  e.printStackTrace();
  } catch (IOException e)
  {
  e.printStackTrace();
  }
  // 取消下载对话框显示
  mDownloadDialog.dismiss();
 }
 }
 
 /**
 * 安装APK文件
 */
 private void installApk()
 {
 File apkfile = new File(mSavePath, mHashMap.get("name"));
 if (!apkfile.exists())
 {
  return;
 }
 Intent i = new Intent(Intent.ACTION_VIEW);
 i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive");
 mContext.startActivity(i);
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 探讨如何真正掌握Java EE,包括所需技能、工具和实践经验。资深软件教学总监李刚分享了对毕业生简历中常见问题的看法,并提供了详尽的标准。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍如何使用arm-eabi-gdb调试Android平台上的C/C++程序。通过具体步骤和实用技巧,帮助开发者更高效地进行调试工作。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 解决MongoDB Compass远程连接问题
    本文记录了在使用阿里云服务器部署MongoDB后,通过MongoDB Compass进行远程连接时遇到的问题及解决方案。详细介绍了从防火墙配置到安全组设置的各个步骤,帮助读者顺利解决问题。 ... [详细]
  • 本文详细探讨了HTTP 500内部服务器错误的成因、解决方案及其在Web开发中的影响。通过对具体案例的分析,帮助读者理解并解决此类问题。 ... [详细]
  • 本文探讨了在通过 API 端点调用时,使用猫鼬(Mongoose)的 findOne 方法总是返回 null 的问题,并提供了详细的解决方案和建议。 ... [详细]
  • 本文介绍如何使用布局文件在Android应用中排列多行TextView和Button,使其占据屏幕的特定比例,并提供示例代码以帮助理解和实现。 ... [详细]
author-avatar
一起去钓鱼
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有