前言:应用内更新现在有各种各样的第三方可供使用,例如:Bugly、蒲公英、友盟等等,只需要接入相关的sdk就可以很简便的实现应用内更新的功能。即使不使用第三方,跟后台配合也可以完成应用内更新的功能,前提是需要有文件服务器存储apk文件。之所以有这篇文章,是因为要满足:1、不使用第三方sdk;2、升级逻辑完全掌握在自己手中;3、没有文件服务器。文章中实现应用内升级主要是拦截WebView网页下载事件。

首先描述一下实现的思路:

  1. 判断是否升级。

    判断服务器返回数据中的版本号与本地版本号做对比,如果需要升级,执行 第二步

  2. 需要升级,判断强制更新、普通更新。

    这里强制更新、普通更新的判断逻辑根据具体业务逻辑实现即可,我提供我实现的方式仅供参考:需要更新时,我在主页会弹出一个自定义的透明Activity,中间有升级内容弹框布局+取消Icon,如果为强制更新,隐藏取消Icon+Activity禁用返回;如果为普通更新,可正常返回。点击更新按钮跳转后的WebView界面亦如此。

  3. 升级,采用WebView拦截下载事件 【重点】

    利用WebView的DownloadListener拦截网页中的下载事件,将网页中的下载事件交给我们自己处理。代码如下:

    WebView.setDownloadListener((url, userAgent, contentDisposition, mimeType, contentLength) -> {
    	// do something,例如:弹出下载进度条、判断文件类型、存储文件、进行安装。
    })
    

本篇文章最重要的就是靠这个监听去拦截用户在蒲公英网页中点击安装的事件,然后做下载的各种处理。

参数解释:1、url:文件url;2、userAgent:HTTP请求头部用来标识客户端信息的字符串;3、contentDisposition:用户想把请求所得的内容存为一个文件的时候提供一个默认的文件名;4、mimeType:下载的文件类型;5、contentLength:下载的文件内容总大小。

实现过程中遇到的问题:
1、WebView加载蒲公英界面底部存在广告 ,这个一般与运营商劫持有关:
Android WebView去除广告
,或者使用大厂的开源WebView方案。

2、下载完成后安装问题。 这个跟系统版本有关,Android 7.0及以上 的版本安装apk需要使用 ContentProvider ,具体代码如下:

java代码:

//安装应用
private void installApk(File apk) {
    Intent intent = new Intent(Intent.ACTION_VIEW);
    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
        intent.setDataAndType(Uri.fromFile(apk), "application/vnd.android.package-archive");
    } else {
        Uri uri = FileProvider.getUriForFile(this, BuildConfig.APPLICATION_ID + ".fileProvider", apk);
        intent.setDataAndType(uri, "application/vnd.android.package-archive");
        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    }
    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    startActivity(intent);
}

同时,还需要在 AndroidManifest 中添加配置:

AndroidManifest.xml 中:

<provider
   android:name="android.support.v4.content.FileProvider"
   android:authorities="${applicationId}.fileProvider"
   android:exported="false"
   android:grantUriPermissions="true">
   <meta-data
       android:name="android.support.FILE_PROVIDER_PATHS"
       android:resource="@xml/provider_paths"/>
</provider>

其中有一个 provider_paths.xml 文件,如下:

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <root-path name="root_path" path="."/>
    <external-path path="." name="external_storage_root" />
    <external-path name="external_files" path="."/>
</paths>

配置过程中一定注意:java代码中的 application_id 要和 AndroidManifest.xmlprovider 节点中的 application_id 一致,我是将 application_id 配置在 build.gradle 中通过BuildConfig暴露出来供外部调用。

另:Android 8.0及以上还需要添加权限:

<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES"/>

补充

之后查看了蒲公英的api文档,提供了很多api给开发者使用,例如:获取最新版本、获取版本记录、下载apk等等,如果需要做应用内更新或者版本历史功能的,完全可以不用后台,直接使用蒲公英的api即可,但是存在一个弊端,就是要完全依赖于蒲公英了。实现一个功能有各种方式,需要自己权衡利弊去选择更适合的方式。