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

AndroidApp实现应用内部自动更新的最基本方法示例

这篇文章主要介绍了实现AndroidApp内部自动更新的最基本方法示例,包括IIS服务器端的简单布置,需要的朋友可以参考下

这只是初步的实现,并没有加入自动编译等功能。需要手动更改更新的xml文件和最新的apk。   
共涉及到四个文件!
一、客户端
AndroidUpdateTestActivity:程序首页
main.xml:首页布局
Update:更新类
softupdate_progress:更新等待界面

Updage

package majier.test; 
 
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.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; 
 
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 Update { 
  private static final int DOWNLOAD = 1; 
  private static final int DOWNLOAD_FINISH = 2; 
  private static final int CONNECT_FAILED = 0; 
  private static final int CONNECT_SUCCESS = 1; 
  HashMap mHashMap; 
  private String mSavePath; 
  private int progress; 
  private boolean cancelUpdate = false; 
  private Context mContext; 
  private ProgressBar mProgress; 
  private Dialog mDownloadDialog; 
  private String mXmlPath; // 服务器更新xml存放地址 
 
  public Update(Context context, String xmlPath, String savePath) { 
    this.mCOntext= context; 
    this.mXmlPath = xmlPath; 
    this.mSavePath = savePath; 
  } 
 
  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 void checkUpdate() { 
    new Thread(new Runnable() { 
      @Override 
      public void run() { 
        try { 
          URL url = new URL(mXmlPath); 
          HttpURLConnection cOnn= (HttpURLConnection) url 
              .openConnection(); 
          conn.setConnectTimeout(5000); 
          InputStream inStream = conn.getInputStream(); 
          mHashMap = parseXml(inStream); 
          Message msg = new Message(); 
          msg.what = CONNECT_SUCCESS; 
          handler.sendMessage(msg); 
        } catch (Exception e) { 
          Message msg = new Message(); 
          msg.what = CONNECT_FAILED; 
          handler.sendMessage(msg); 
        } 
      } 
    }).run(); 
  } 
 
  /** 
   * 访问服务器更新XML 
   */ 
  Handler handler = new Handler() { 
    @Override 
    public void handleMessage(Message msg) { 
      super.handleMessage(msg); 
      switch (msg.what) { 
      case CONNECT_FAILED: 
        Toast.makeText(mContext, "访问服务器失败!", Toast.LENGTH_SHORT).show(); 
        break; 
      case CONNECT_SUCCESS: 
        if (null != mHashMap) { 
          int serviceCode = Integer.valueOf(mHashMap.get("version")); 
          if (serviceCode > getVersionCode(mContext)) { 
            showNoticeDialog(); 
          } 
        } 
        break; 
      } 
    } 
  }; 
 
  /** 
   * 获取程序版本号 
   */ 
  private int getVersionCode(Context context) { 
    int versiOnCode= 0; 
    try { 
      versiOnCode= context.getPackageManager().getPackageInfo( 
          mContext.getPackageName(), 0).versionCode; 
    } catch (NameNotFoundException e) { 
      e.printStackTrace(); 
    } 
    return versionCode; 
  } 
 
  /** 
   * 是否更新提示窗口 
   */ 
  private void showNoticeDialog() { 
    AlertDialog.Builder builder = new Builder(mContext); 
    builder.setTitle("软件更新"); 
    builder.setMessage("检测到新版本,是否更新?"); 
    builder.setPositiveButton("更新", 
        new OnClickListener() { 
          @Override 
          public void onClick(DialogInterface dialog, int which) { 
            dialog.dismiss(); 
            showDownloadDialog(); 
          } 
        }); 
 
    builder.setNegativeButton("取消", 
        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("正在更新"); 
    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("取消下载", 
        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(); 
  } 
 
  /** 
   * 下载程序 
   */ 
  private class downloadApkThread extends Thread { 
    @Override 
    public void run() { 
      try { 
        if (Environment.getExternalStorageState().equals( 
            Environment.MEDIA_MOUNTED)) { 
 
          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); 
  } 
 
  private 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 

AndroidUpdateTestActivity

package majier.test; 
 
import android.app.Activity; 
import android.os.Bundle; 
import android.os.Environment; 
 
public class AndroidUpdateTestActivity extends Activity { 
  /** Called when the activity is first created. */ 
  @Override 
  public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.main); 
    update(); 
  } 
   
  private void update() { 
    String sdpath = Environment.getExternalStorageDirectory() + "/"; 
    String mSavePath = sdpath + "boiler/"; 
    Update updateManager = new Update(this, 
        "http://localhost:8011/abcd.xml", mSavePath); 
    updateManager.checkUpdate(); 
  } 
} 

main.xml

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
 
   
 
 

softupdate_progress.xml

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
   
 

每次生成新的apk前,需要修改系统的版本号。

2016324165959773.png (747×620)

修改version code 和version name。上面的代码可以看出,系统是根据version code来判断是否需要更新的。version name作为一个版本名称。
这里我建议version code从10开始,这样方面名称修改(1.1、1.2)。
修改完成后,生成系统。然后将apk文件放在服务端的文件下。

二、服务端
服务端主要是建立一个网址供用户下载apk。在IIS上新建网站

http://localhost:8011/。将更新文件和更新的xml放在目录下。 

2016324170029494.png (663×72)

version.xml格式

 
  12 
  BoilerAndroid_1.1 
  http://192.168.0.33:8011/boilerandroid.apk 
 

version对应着新程序的version code;
name随便起名;
url对应apk的下载路径。

在这里有可能会遇见一个问题,访问url路径时IIS报错。主要是因为IIS并不认识apk,不知道如何处理。
这里我们在IIS中新增安卓程序的MIME类型,来使apk支持下载。
在“IIS管理器”中查看所建立的网站——MIME类型——添加。
文件扩展名:.apk
MIME类型:application/vnd.android.package-archive

2016324170048657.png (360×222)

这样就可以下载了。
目前只是一个简单的自动更新程序。我们可以看出,其中版本号需要自己填写,而且要与xml中的对应,apk需要生成后放在更新网址下。
这么的人为操作,很容易造成失误。因此,接下来我们要研究下自动发布更新版本,并且版本号与svn对应,在提交svn后,自动改变程序的版本号。


推荐阅读
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Java源代码安全审计(二):使用Fortify-sca工具进行maven项目安全审计
    本文介绍了使用Fortify-sca工具对maven项目进行安全审计的过程。作者通过对Fortify的研究和实践,记录了解决问题的学习过程。文章详细介绍了maven项目的处理流程,包括clean、build、Analyze和Report。在安装mvn后,作者遇到了一些错误,并通过Google和Stack Overflow等资源找到了解决方法。作者分享了将一段代码添加到pom.xml中的经验,并成功进行了mvn install。 ... [详细]
  • Maven入门、什么是Maven、如何使用Maven、Maven的项目结构、简单的Mavenjava项目、Maven常用命令、Maven项目之间的引用、Maven依赖的传递、可选、排除day01
    目录第一节Maven入门1.1什么是Maven1.2如何使用Maven第一步:下载Maven第二步:配置Maven的环境变量第三步:了解什 ... [详细]
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • 本文介绍了在Linux下安装和配置Kafka的方法,包括安装JDK、下载和解压Kafka、配置Kafka的参数,以及配置Kafka的日志目录、服务器IP和日志存放路径等。同时还提供了单机配置部署的方法和zookeeper地址和端口的配置。通过实操成功的案例,帮助读者快速完成Kafka的安装和配置。 ... [详细]
  • 本文是关于自学Android的笔记,包括查看类的源码的方法,活动注册的必要性以及布局练习的重要性。通过学习本文,读者可以了解到在自学Android过程中的一些关键点和注意事项。 ... [详细]
  • 本文介绍了使用cacti监控mssql 2005运行资源情况的操作步骤,包括安装必要的工具和驱动,测试mssql的连接,配置监控脚本等。通过php连接mssql来获取SQL 2005性能计算器的值,实现对mssql的监控。详细的操作步骤和代码请参考附件。 ... [详细]
  • svnWebUI:一款现代化的svn服务端管理软件
    svnWebUI是一款图形化管理服务端Subversion的配置工具,适用于非程序员使用。它解决了svn用户和权限配置繁琐且不便的问题,提供了现代化的web界面,让svn服务端管理变得轻松。演示地址:http://svn.nginxwebui.cn:6060。 ... [详细]
  • 本文介绍了在Ubuntu下制作deb安装包及离线安装包的方法,通过备份/var/cache/apt/archives文件夹中的安装包,并建立包列表及依赖信息文件,添加本地源,更新源列表,可以在没有网络的情况下更新系统。同时提供了命令示例和资源下载链接。 ... [详细]
  • Jquery 跨域问题
    为什么80%的码农都做不了架构师?JQuery1.2后getJSON方法支持跨域读取json数据,原理是利用一个叫做jsonp的概念。当然 ... [详细]
  • 云原生应用最佳开发实践之十二原则(12factor)
    目录简介一、基准代码二、依赖三、配置四、后端配置五、构建、发布、运行六、进程七、端口绑定八、并发九、易处理十、开发与线上环境等价十一、日志十二、进程管理当 ... [详细]
  • 1jdk去网站下载,然后拷贝到linux上;或直接wgethttp:download.oracle.comotn-pubjavajdk8u181-b1 ... [详细]
  • SVN安装配置和使用
    简介:SVN是Subversion的简称,是一个开放源代码的版本控制系统,相较于RCS、CVS,它采用了分支管理系统,它的设计目标就是取代CVS。互联网上很多版本控制服务已从CVS ... [详细]
author-avatar
郑郑郑克_583
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有