app升级是每一个android应用的标配了,大部分应用都会有升级提醒和apk下载安装(如果系统允许静默安装估计就没有提醒这段了)。
以前的升级是自己写http下载或者通过系统提供的DownloadManager进行下载。无意间发现bugly提供下载更新服务而且免费(后来又提供了热更新),就乐呵呵的接入了sdk,毕竟是大厂的东西,值得信赖。开始用的时候用的很稳定(估计是手机厂商还没用上牛轧糖),后来就有人反应更新失败,一直提示安装,开始有些不相信直接让用户卸载重装(还好用户是公司自己人,是外人的话那就有卸载无安装了)。
检查问题的时候也看了一下bugly的接入指南,加入了android7.0的适配。开始觉得7.0的用户还少而且我的编译版本是5.0就没做适配。当最近更新反应更新失败的用户比较多时就仔细查了一下问题。也看了一下鹅厂工程师的博客(http://blog.csdn.net/wwj_748/article/details/52664647)。原来这是由于Android7.0执行了“StrictMode API 政策禁”的原因。" StrictMode API 政策" 是指禁止向你的应用外公开 file:// URI。 如果一项包含文件 file:// URI类型 的 Intent 离开你的应用,应用失败,并出现 FileUriExposedException 异常。
以前更新通过隐式调用系统安装程序
1
2
3
4
|
Intent i =
new
Intent(Intent.ACTION_VIEW);
i.setDataAndType(Uri.fromFile(file),
"application/vnd.android.package-archive"
);
i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(i);
|
这段代码是通过intent设置数据和类型,然后通过context在新的task中启动安装apk的程序。
我们看到intent设置数据时,传递的是一个Uri,这个在API<24是没有问题的,但在Android N已经禁止你对外公开file://URI.所以我们SDK的问题就出自Uri.fromFile(file)获取uri的时候。
Android N已经给出明确解决方案,如果你的程序需要在应用间共享文件,您应发送一项 content://URI,并授予 URI 临时访问权限。进行此授权的最简单方式是使用 FileProvider类。
首先在AndroidManifest中注册FileProvider
代码示例:
1
2
3
4
5
6
7
8
9
|
<
provider
android:name
=
"android.support.v4.content.FileProvider"
android:authorities
=
"com.bugly.upgrade.demo.fileProvider"
android:exported
=
"false"
android:grantUriPermissions
=
"true"
>
<
meta-data
android:name
=
"android.support.FILE_PROVIDER_PATHS"
android:resource
=
"@xml/provider_paths"
/>
</
provider
>
|
这里要注意一下,FileProvider是support-v4包里面的,所以在你的程序必须要引入support-v4包。
我们可以看到在provider中需要配置相应的meta-data,这个是共享文件的路径,在res目录下新建xml文件夹并新建对应的xml文件(如下面的provider_paths),如下所示:
1
2
3
4
5
6
7
|
<?
xml
version
=
"1.0"
encoding
=
"utf-8"
?>
<
paths
xmlns:android
=
"http://schemas.android.com/apk/res/android"
>
<!-- /storage/emulated/0/Download/-->
<
external-path
name
=
"beta_external_apk"
path
=
"Download/"
/>
<!--/storage/emulated/0/Android/data/-->
<
external-path
name
=
"beta_external_apk2"
path
=
"Android/data//"
/>
</
paths
>
|
name表示一个URI路径段,path表示指定要分享路径的子目录。这样系统安装程序就能访问下载的apk文件了。
由于手里没有7.0的设备在优测上租个设备(还好有一个小时的试用)测试了一下,完美解决。谷歌系统升级竟然不向下兼容,这也是个坑。
编程路上真是一道坎一道坑。愿世界永无bug,阿门。