app升级是每一个android应用的标配了,大部分应用都会有升级提醒和apk下载安装(如果系统允许静默安装估计就没有提醒这段了)。
以前的升级是自己写http下载或者通过系统提供的DownloadManager进行下载。无意间发现bugly提供下载更新服务而且免费(后来又提供了热更新),就乐呵呵的接入了sdk,毕竟是大厂的东西,值得信赖。开始用的时候用的很稳定(估计是手机厂商还没用上牛轧糖),后来就有人反应更新失败,一直提示安装,开始有些不相信直接让用户卸载重装(还好用户是公司自己人,是外人的话那就有卸载无安装了)。
检查问题的时候也看了一下bugly的接入指南,加入了android7.0的适配。开始觉得7.0的用户还少而且我的编译版本是5.0就没做适配。当最近更新反应更新失败的用户比较多时就仔细查了一下问题。也看了一下鹅厂工程师的博客()。原来这是由于Android7.0执行了“StrictMode API 政策禁”的原因。" StrictMode API 政策" 是指禁止向你的应用外公开 file:// URI。 如果一项包含文件 file:// URI类型 的 Intent 离开你的应用,应用失败,并出现 FileUriExposedException 异常。
以前更新通过隐式调用系统安装程序
|
这段代码是通过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
代码示例:
|
这里要注意一下,FileProvider是support-v4包里面的,所以在你的程序必须要引入support-v4包。
我们可以看到在provider中需要配置相应的meta-data,这个是共享文件的路径,在res目录下新建xml文件夹并新建对应的xml文件(如下面的provider_paths),如下所示:
|
name表示一个URI路径段,path表示指定要分享路径的子目录。这样系统安装程序就能访问下载的apk文件了。
由于手里没有7.0的设备在优测上租个设备(还好有一个小时的试用)测试了一下,完美解决。谷歌系统升级竟然不向下兼容,这也是个坑。