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

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

由于Android项目开源所致,市面上出现了N多安卓软件市场。为了让我们开发的软件有更多的用户使用,我们需要向N多市场发布,软件升级后,我们

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

既然实现自动更新,我们首先必须让我们的应用知道是否存在新版本的软件,因此我们可以在自己的网站上放置配置文件,存放软件的版本信息:

 
  2 
  baidu_xinwen_1.1.0 
  http://gdown.baidu.com/data/wisegame/f98d235e39e29031/baiduxinwen.apk 
 

在这里我使用的是XML文件,方便读取。由于XML文件内容比较少,因此可通过DOM方式进行文件的解析:

public class ParseXmlService 
{ 
  public HashMap parseXml(InputStream inStream) throws Exception 
  { 
    HashMap hashMap = new HashMap(); 
     
    // 实例化一个文档构建器工厂 
    DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); 
    // 通过文档构建器工厂获取一个文档构建器 
    DocumentBuilder builder = factory.newDocumentBuilder(); 
    // 通过文档通过文档构建器构建一个文档实例 
    Document document = builder.parse(inStream); 
    //获取XML文件根节点 
    Element root = document.getDocumentElement(); 
    //获得所有子节点 
    NodeList childNodes = root.getChildNodes(); 
    for (int j = 0; j 

 通过parseXml()方法,我们可以获取服务器上应用的版本、文件名以及下载地址。紧接着我们就需要获取到我们手机上应用的版本信息:

 /** 
 * 获取软件版本号 
 * 
 * @param context 
 * @return 
 */ 
private int getVersionCode(Context context) 
{ 
  int versiOnCode= 0; 
  try 
  { 
    // 获取软件版本号, 
    versiOnCode= context.getPackageManager().getPackageInfo("com.szy.update", 0).versionCode; 
  } catch (NameNotFoundException e) 
  { 
    e.printStackTrace(); 
  } 
  return versionCode; 
} 

通过该方法我们获取到的versionCode对应AndroidManifest.xml下android:versionCode。android:versionCode和android:versionName两个属性分别表示版本号,版本名称。versionCode是整数型,而versionName是字符串。由于versionName是给用户看的,不太容易比较大小,升级检查时,就可以检查versionCode。把获取到的手机上应用版本与服务器端的版本进行比较,应用就可以判断处是否需要更新软件。

处理流程

处理代码

package com.szy.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 android.app.AlertDialog; 
import android.app.Dialog; 
import android.app.AlertDialog.Builder; 
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.os.Message; 
import android.view.LayoutInflater; 
import android.view.View; 
import android.widget.ProgressBar; 
import android.widget.Toast; 
 

 
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 mProgress; 
  private Dialog mDownloadDialog; 
 
  private Handler mHandler = new Handler() 
  { 
    public void handleMessage(Message msg) 
    { 
      switch (msg.what) 
      { 
      // 正在下载 
      case DOWNLOAD: 
        // 设置进度条位置 
        mProgress.setProgress(progress); 
        break; 
      case DOWNLOAD_FINISH: 
        // 安装文件 
        installApk(); 
        break; 
      default: 
        break; 
      } 
    }; 
  }; 
 
  public UpdateManager(Context context) 
  { 
    this.mCOntext= context; 
  } 
 
  /** 
   * 检测软件更新 
   */ 
  public void checkUpdate() 
  { 
    if (isUpdate()) 
    { 
      // 显示提示对话框 
      showNoticeDialog(); 
    } else 
    { 
      Toast.makeText(mContext, R.string.soft_update_no, Toast.LENGTH_LONG).show(); 
    } 
  } 
 
  /** 
   * 检查软件是否有更新版本 
   * 
   * @return 
   */ 
  private 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) 
    { 
      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) 
{ 
  int versiOnCode= 0; 
  try 
  { 
    // 获取软件版本号,对应AndroidManifest.xml下android:versionCode 
    versiOnCode= context.getPackageManager().getPackageInfo("com.szy.update", 0).versionCode; 
  } catch (NameNotFoundException e) 
  { 
    e.printStackTrace(); 
  } 
  return versionCode; 
} 
 
  /** 
   * 显示软件更新对话框 
   */ 
  private void showNoticeDialog() 
  { 
    // 构造对话框 
    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) 
      { 
        dialog.dismiss(); 
        // 显示下载对话框 
        showDownloadDialog(); 
      } 
    }); 
    // 稍后更新 
    builder.setNegativeButton(R.string.soft_update_later, new OnClickListener() 
    { 
      @Override 
      public void onClick(DialogInterface dialog, int which) 
      { 
        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 v = inflater.inflate(R.layout.softupdate_progress, null); 
    mProgress = (ProgressBar) v.findViewById(R.id.update_progress); 
    builder.setView(v); 
    // 取消更新 
    builder.setNegativeButton(R.string.soft_update_cancel, new OnClickListener() 
    { 
      @Override 
      public void onClick(DialogInterface dialog, int which) 
      { 
        dialog.dismiss(); 
        // 设置取消状态 
        cancelUpdate = true; 
      } 
    }); 
    mDownloadDialog = builder.create(); 
    mDownloadDialog.show(); 
    // 现在文件 
    downloadApk(); 
  } 
 
  /** 
   * 下载apk文件 
   */ 
  private void downloadApk() 
  { 
    // 启动新线程下载软件 
    new downloadApkThread().start(); 
  } 
 
  /** 
   * 下载文件线程 
   * 
   * @author coolszy 
   */ 
  private class downloadApkThread extends Thread 
  { 
    @Override 
    public void run() 
    { 
      try 
      { 
        // 判断SD卡是否存在,并且是否具有读写权限 
        if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) 
        { 
          // 获得存储卡的路径 
          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安装APK文件 
    Intent i = new Intent(Intent.ACTION_VIEW); 
    i.setDataAndType(Uri.parse("file://" + apkfile.toString()), "application/vnd.android.package-archive"); 
    mContext.startActivity(i); 
  } 
} 

效果图

检查模拟器SDCARD是否存在下载文件:
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 本文介绍了使用AJAX的POST请求实现数据修改功能的方法。通过ajax-post技术,可以实现在输入某个id后,通过ajax技术调用post.jsp修改具有该id记录的姓名的值。文章还提到了AJAX的概念和作用,以及使用async参数和open()方法的注意事项。同时强调了不推荐使用async=false的情况,并解释了JavaScript等待服务器响应的机制。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • HDFS2.x新特性
    一、集群间数据拷贝scp实现两个远程主机之间的文件复制scp-rhello.txtroothadoop103:useratguiguhello.txt推pushscp-rr ... [详细]
  • Android系统移植与调试之如何修改Android设备状态条上音量加减键在横竖屏切换的时候的显示于隐藏
    本文介绍了如何修改Android设备状态条上音量加减键在横竖屏切换时的显示与隐藏。通过修改系统文件system_bar.xml实现了该功能,并分享了解决思路和经验。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • 本文介绍了在Linux下安装和配置Kafka的方法,包括安装JDK、下载和解压Kafka、配置Kafka的参数,以及配置Kafka的日志目录、服务器IP和日志存放路径等。同时还提供了单机配置部署的方法和zookeeper地址和端口的配置。通过实操成功的案例,帮助读者快速完成Kafka的安装和配置。 ... [详细]
  • 本文是关于自学Android的笔记,包括查看类的源码的方法,活动注册的必要性以及布局练习的重要性。通过学习本文,读者可以了解到在自学Android过程中的一些关键点和注意事项。 ... [详细]
author-avatar
mobiledu2502875007
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有