`

转:Android应用的自动升级、更新模块的实现

阅读更多
Android应用的自动升级、更新模块的实现



我们看到很多Android应用都具有自动更新功能,用户一键就可以完成软件的升级更新。得益于Android系统的软件包管理和安装机制,这一功能实现起来相当简单,下面我们就来实践一下。首先给出界面效果:


1. 准备知识
在AndroidManifest.xml里定义了每个Android apk的版本标识:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="com.myapp"
      android:versionCode="1"
      android:versionName="1.0.0">
<application></application>
</manifest>
复制代码
其中,android:versionCode和android:versionName两个字段分别表示版本代码,版本名称。versionCode是整型数字,versionName是字符串。由于version是给用户看的,不太容易比较大小,升级检查时,可以以检查versionCode为主,方便比较出版本的前后大小。
那么,在应用中如何读取AndroidManifest.xml中的versionCode和versionName呢?可以使用PackageManager的API,参考以下代码:
public static int getVerCode(Context context) {
        int verCode = -1;
        try {
            verCode = context.getPackageManager().getPackageInfo(
                    "com.myapp", 0).versionCode;
        } catch (NameNotFoundException e) {
            Log.e(TAG, e.getMessage());
        }
        return verCode;
    }
  
    public static String getVerName(Context context) {
        String verName = "";
        try {
            verName = context.getPackageManager().getPackageInfo(
                    "com.myapp", 0).versionName;
        } catch (NameNotFoundException e) {
            Log.e(TAG, e.getMessage());
        }
        return verName;  
}
复制代码
或者在AndroidManifest中将android:versionName="1.2.0"写成android:versionName="@string/app_versionName",然后在values/strings.xml中添加对应字符串,这样实现之后,就可以使用如下代码获得版本名称:
public static String getVerName(Context context) {
        String verName = context.getResources()
        .getText(R.string.app_versionName).toString();
        return verName;
}
复制代码
同理,apk的应用名称可以这样获得:
public static String getAppName(Context context) {
        String verName = context.getResources()
        .getText(R.string.app_name).toString();
        return verName;
}
复制代码
2. 流程框架

3. 版本检查
在服务端放置最新版本的apk文件,如:http://localhost/myapp/myapp.apk
同时,在服务端放置对应此apk的版本信息调用接口或者文件,如:http://localhost/myapp/ver.json
ver.json中的内容为:
[{"appname":"jtapp12","apkname":"jtapp-12-updateapksamples.apk","verName":1.0.1,"verCode":2}]
复制代码
然后,在手机客户端上进行版本读取和检查:
private boolean getServerVer () {
        try {
            String verjson = NetworkTool.getContent(Config.UPDATE_SERVER
                    + Config.UPDATE_VERJSON);
            JSONArray array = new JSONArray(verjson);
            if (array.length() > 0) {
                JSONObject obj = array.getJSONObject(0);
                try {
                    newVerCode = Integer.parseInt(obj.getString("verCode"));
                    newVerName = obj.getString("verName");
                } catch (Exception e) {
                    newVerCode = -1;
                    newVerName = "";
                    return false;
                }
            }
        } catch (Exception e) {
            Log.e(TAG, e.getMessage());
            return false;
        }
        return true;
    }
复制代码
比较服务器和客户端的版本,并进行更新操作。
   if (getServerVerCode()) {
            int vercode = Config.getVerCode(this); // 用到前面第一节写的方法
            if (newVerCode > vercode) {
                doNewVersionUpdate(); // 更新新版本
            } else {
                notNewVersionShow(); // 提示当前为最新版本
            }
        }       
复制代码
详细方法:
private void notNewVersionShow() {
                int verCode = Config.getVerCode(this);
                String verName = Config.getVerName(this);
                StringBuffer sb = new StringBuffer();
                sb.append("当前版本:");
                sb.append(verName);
                sb.append(" Code:");
                sb.append(verCode);
                sb.append(",/n已是最新版,无需更新!");
                Dialog dialog = new AlertDialog.Builder(Update.this).setTitle("软件更新")
                                .setMessage(sb.toString())// 设置内容
                                .setPositiveButton("确定",// 设置确定按钮
                                                new DialogInterface.OnClickListener() {
                                                        @Override
                                                        public void onClick(DialogInterface dialog,
                                                                        int which) {
                                                                finish();
                                                        }
                                                }).create();// 创建
                // 显示对话框
                dialog.show();
        }
        private void doNewVersionUpdate() {
                int verCode = Config.getVerCode(this);
                String verName = Config.getVerName(this);
                StringBuffer sb = new StringBuffer();
                sb.append("当前版本:");
                sb.append(verName);
                sb.append(" Code:");
                sb.append(verCode);
                sb.append(", 发现新版本:");
                sb.append(newVerName);
                sb.append(" Code:");
                sb.append(newVerCode);
                sb.append(", 是否更新?");
                Dialog dialog = new AlertDialog.Builder(Update.this)
                                .setTitle("软件更新")
                                .setMessage(sb.toString())
                                // 设置内容
                                .setPositiveButton("更新",// 设置确定按钮
                                                new DialogInterface.OnClickListener() {
                                                        @Override
                                                        public void onClick(DialogInterface dialog,
                                                                        int which) {
                                                                pBar = new ProgressDialog(Update.this);
                                                                pBar.setTitle("正在下载");
                                                                pBar.setMessage("请稍候...");
                                                                pBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);
                                                                downFile(Config.UPDATE_SERVER + Config.UPDATE_APKNAME);
                                                        }
                                                })
                                .setNegativeButton("暂不更新",
                                                new DialogInterface.OnClickListener() {
                                                        public void onClick(DialogInterface dialog,
                                                                        int whichButton) {
                                                                // 点击"取消"按钮之后退出程序
                                                                finish();
                                                        }
                                                }).create();// 创建
                // 显示对话框
                dialog.show();
        }
复制代码
4. 下载模块
注,本部分参考了前人的相关实现,
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;

import android.app.AlertDialog;
import android.app.Dialog;
import android.app.ProgressDialog;
import android.content.DialogInterface;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;

public class Update extends BaseActivity {
public ProgressDialog pBar;
private Handler handler = new Handler();

@Override
protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.update);
  Dialog dialog = new AlertDialog.Builder(Update.this).setTitle("系统更新")
    .setMessage("发现新版本,请更新!")// 设置内容
    .setPositiveButton("确定",// 设置确定按钮
      new DialogInterface.OnClickListener() {

       @Override
       public void onClick(DialogInterface dialog,
         int which) {
        pBar = new ProgressDialog(Update.this);
        pBar.setTitle("正在下载");
        pBar.setMessage("请稍候...");
        pBar
          .setProgressStyle(ProgressDialog.STYLE_SPINNER);
        downFile("http://url:8765/OA.apk");
       

       }

      }).setNegativeButton("取消",
      new DialogInterface.OnClickListener() {
       public void onClick(DialogInterface dialog,
         int whichButton) {
        // 点击"取消"按钮之后退出程序
       
       }
      }).create();// 创建
  // 显示对话框
  dialog.show();

}

void downFile(final String url) {
  pBar.show();
  new Thread() {
   public void run() {
    HttpClient client = new DefaultHttpClient();
    // params[0]代表连接的url
    HttpGet get = new HttpGet(url);
    HttpResponse response;
    try {
     response = client.execute(get);
     HttpEntity entity = response.getEntity();
     long length = entity.getContentLength();
     InputStream is = entity.getContent();
     FileOutputStream fileOutputStream = null;
     if (is != null) {

      File file = new File(Environment
        .getExternalStorageDirectory(), "OA.apk");
      fileOutputStream = new FileOutputStream(file);
     
      byte[] buf = new byte[1024];
      int ch = -1;
      int count = 0;
      while ((ch = is.read(buf)) != -1) {
       // baos.write(buf, 0, ch);
       fileOutputStream.write(buf, 0, ch);
       count += ch;
       if (length > 0) {
      
       }

      }

     }
     fileOutputStream.flush();
     if (fileOutputStream != null) {
      fileOutputStream.close();
     }
     down();
    } catch (ClientProtocolException e) {
     // TODO Auto-generated catch block
     e.printStackTrace();
    } catch (IOException e) {
     // TODO Auto-generated catch block
     e.printStackTrace();
    }
   }

  }.start();

}

void down() {
  handler.post(new Runnable() {
   public void run() {
    pBar.cancel();
    update();
   }
  });

}

void update() {

  Intent intent = new Intent(Intent.ACTION_VIEW);
  intent.setDataAndType(Uri.fromFile(new File("/sdcard/OA.apk")),
    "application/vnd.android.package-archive");
  startActivity(intent);
}

}
复制代码

```````````````````````
    void downFile(final String url) {
        pBar.show();
        new Thread() {
            public void run() {
                HttpClient client = new DefaultHttpClient();
                HttpGet get = new HttpGet(url);
                HttpResponse response;
                try {
                    response = client.execute(get);
                    HttpEntity entity = response.getEntity();
                    long length = entity.getContentLength();
                    InputStream is = entity.getContent();
                    FileOutputStream fileOutputStream = null;
                    if (is != null) {
                        File file = new File(
                                Environment.getExternalStorageDirectory(),
                                Config.UPDATE_SAVENAME);
                        fileOutputStream = new FileOutputStream(file);
                        byte[] buf = new byte[1024];
                        int ch = -1;
                        int count = 0;
                        while ((ch = is.read(buf)) != -1) {
                            fileOutputStream.write(buf, 0, ch);
                            count += ch;
                            if (length > 0) {
                            }
                        }
                    }
                    fileOutputStream.flush();
                    if (fileOutputStream != null) {
                        fileOutputStream.close();
                    }
                    down();
                } catch (ClientProtocolException e) {
                    e.printStackTrace();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }.start();
    }
复制代码
下载完成,通过handler通知主ui线程将下载对话框取消。
void down() {
        handler.post(new Runnable() {
            public void run() {
                pBar.cancel();
                update();
            }
        });
}
复制代码
5. 安装应用
    void update() {
        Intent intent = new Intent(Intent.ACTION_VIEW);
        intent.setDataAndType(Uri.fromFile(new File(Environment
                .getExternalStorageDirectory(), Config.UPDATE_SAVENAME)),
                "application/vnd.android.package-archive");
        startActivity(intent);
    }
复制代码
果你将apk应用发布到market上,那么,你会发现market内建了类似的模块,可以自动更新或者提醒你是否更新应用。那么,对于你自己的应用需要自动更新的话,自己内建一个是不是更加方便了呢?本文提到的代码大多是在UpdateActivity.java中实现,为了能够使更新过程更加友好,可以在最初launcher的Activity中建立一个线程,用来检查服务端是否有更新。有更新的时候就启动UpdateActivity,这样的使用体验更加平滑。

本文例程源码查看/下载:
分享到:
评论

相关推荐

    转帖:Android应用的自动升级、更新模块的实现docx.docx

    在Android应用开发中,自动升级和更新功能是一个重要的特性,它允许用户无缝地获取应用的最新版本,提高用户体验。本文将详细介绍如何实现这样一个模块。 **一、基础概念** 1. **版本标识**: 在AndroidManifest.xml...

    Android应用的自动升级、更新模块的实现 .

    总结一下,Android应用的自动升级和更新模块的实现涉及到网络请求、版本比较、文件下载、权限管理、安装逻辑等多个方面。通过合理的代码结构和第三方库的使用,可以高效且安全地实现这一功能,为用户提供便捷的更新...

    Android更新 升级模块

    - 自定义升级模块:有时,系统自带的升级机制可能无法满足特定需求,开发者可以自定义升级模块,如添加断点续传、多线程下载等功能。 4. **安全考虑**: - 更新包的签名验证:确保更新来自可信源,防止恶意软件。...

    Android应用程序的自动更新升级(自身升级、通过tomcat)-IT计算机-毕业设计.zip

    通过实践这个项目,你不仅可以掌握Android应用自动更新的实现,还能提升服务器端的部署和管理技能。此外,此项目还涉及到源码阅读和分析,对于理解Android应用生命周期、网络通信、文件操作等基础知识点也非常有帮助...

    Android-Android实现应用的增量更新和升级

    在Android应用开发中,为了提供更好的用户体验和节省用户流量,增量更新和升级是必不可少的功能。增量更新只下载自上次更新以来发生变化的部分文件,而全量更新则需要下载整个APK文件。本文将深入探讨如何在Android...

    app 升级模块 自动升级

    在移动应用开发中,"app升级模块 自动升级"是一个重要的功能,它允许应用程序自动检测新版本,并在用户同意后无缝地下载和安装更新。这个功能对于保持应用的最新状态和修复潜在问题至关重要。以下是对这个主题的详细...

    Android程序自动更新功能模块的实现方法【附完整demo源码下载】

    在Android应用程序开发中,实现程序自动更新功能是提高用户体验和确保应用安全的重要手段。这个功能允许应用程序在后台检测到新版本时,自动或提示用户进行更新,以获取最新的功能和修复已知问题。以下是一个关于...

    Android-androd升级模块

    `Android-Androd升级模块`,即`ApkUpdateTool`,是一个专为Android应用设计的版本更新框架。这个框架的目标是简化版本更新的实现过程,让开发者能够快速集成到自己的应用中,同时提供了自定义对话框提示和自定义功能...

    Android应用性能优化

    ### Android应用性能优化 在移动开发领域,特别是针对Android平台的应用开发过程中,性能优化是确保应用流畅运行、提升用户体验的关键环节。本篇文章将围绕“Android应用性能优化”这一主题,详细探讨Android应用...

    qt for android 更新APP

    在"qt for android 更新APP"的场景下,我们讨论的是如何在Android应用程序内部实现更新功能。这通常涉及到以下几个关键知识点: 1. **自动检查更新**:应用启动时或在设定的时间间隔内,通过网络请求(通常是HTTP或...

    android版本升级模块

    本文将详细解析一个名为"android版本升级模块"的组件,该组件被设计来简化版本升级的集成过程,使得开发者只需要一行代码就能实现版本检查和更新。 首先,我们关注的是"封装了版本升级功能"这一特点。这意味着这个...

    android 动态加载之自动升级(插件式开发)

    在Android应用开发中,动态加载和插件式开发是一种高级技术,它允许应用程序在运行时加载新的功能或更新,而无需用户重新安装整个应用。这种技术对于实现自动升级和提高用户体验具有重要意义。以下是对这个"android ...

    Android应用架构设计

    良好的架构设计能够使代码更加模块化,降低耦合度,便于后期的迭代和升级。以下是一些关于Android应用架构设计的关键知识点: 1. **分层架构**:常见的分层架构包括表现层(Presentation Layer)、业务逻辑层...

    Android升级更新实例,客户端+服务端

    在Android应用开发中,定期对应用进行升级和更新是保持软件功能完善和安全的重要环节。本实例将探讨如何实现客户端和服务端的协同更新机制,确保用户可以顺利接收并安装最新的应用版本。 一、客户端更新流程 1. **...

    app应用自动&手动更新demo小案例

    "app应用自动&手动更新demo小案例" 提供了一个很好的实践示例,涵盖了如何在Android或iOS平台上实现这两种更新机制。下面我们将深入探讨这两种更新方式及其背后的原理。 首先,手动更新是指用户主动触发应用检查并...

    Android应用自动更新功能实现的方法

    本文给大家分享Android里应用版本更新功能这一块的实现。 一个好的应用软件都是需要好的维护,从初出版本到最后精品,这个过程需要版本不停的更新,那么如何让用户第一时间获取最新的应用安装包呢?那么就要求我们从...

    Android版本更新+附源代码

    在Android应用开发中,版本更新是一项重要的维护工作,它确保用户能够及时获取到最新的功能修复、性能优化和安全更新。本资源提供了一种基于XML配置的Android版本更新方案,简化了更新过程。以下是对这个主题的详细...

    android 系统升级apk

    对于非系统应用,有时可以采用热更新技术,如插件化框架(如Dexposed、DynamicApk等),无需完整升级整个APK,只需更新部分功能模块。 4. **系统更新**: 当涉及到系统级别的APK(如System Apps)升级,过程更为...

Global site tag (gtag.js) - Google Analytics