android7.0更新安装apk的方法

举报
yd_221104950 发表于 2020/12/02 23:45:05 2020/12/02
【摘要】 StrictMode模式 从android7.0开始强制启用StrictMode“严苛模式”。StrictMode是在android2.3引进的类。当时它的作用是作为一个开发工具用的,开发者可以开发者选项中打开它,它可以捕捉到在主线程序发生的磁盘IO读写、网络访问发生的意外,通常这个意外都是ANR(android not response),当时可以用这个工具检测出这...

StrictMode模式

从android7.0开始强制启用StrictMode“严苛模式”。StrictMode是在android2.3引进的类。当时它的作用是作为一个开发工具用的,开发者可以开发者选项中打开它,它可以捕捉到在主线程序发生的磁盘IO读写、网络访问发生的意外,通常这个意外都是ANR(android not response),当时可以用这个工具检测出这些意外,代码就可以做调整:将磁盘读写、网络访问等耗时操作写到非主线程中,以提供更好的体验。但是在android7.0以后,这项功能被强制使用了。

在App之间共享文件

对于android7.0以后的系统,android框架强制使用StrictMode API 策略。这个策略禁止在app外暴露 “file://“URI。如果一个intent包含一个文件URI(以file://开头)离开你的app,那么这个app就报告FileUriExposedException异常。安装apk的功能实际上是系统提供的。Intent会带着我们的意图(包括apk的位置信息)离开我们的app,进入系统中,让系统中的应用来处理。这意味着apk文件需要在app间共享。
为了与其他应用共享文件,你应该发送"content://"URI ,并授予临时访问权限。授予这个临时访问权限的最签单方法就是使用FileProvider类。

注意:当然少不了android6.0之后要动态申请权限这一步。

使用FileProvider辅助完成安装任务

第一步:在manifest.xml申请以下权限,特别是第二个权限,如果缺少的话,在android8.0的手机上安装apk会失败

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

  
 
  • 1
  • 2

第二步:动态申请android.permission.READ_EXTERNAL_STORAGE权限。在android6.0之后,这个权限要动态申请,具体请看Github上的代码
第三步:在manifest.xml中声明FileProvider:

<?xml version="1.0" encoding="utf-8"?>
<manifest> <application> ...... <provider android:name="android.support.v4.content.FileProvider" android:authorities="com.wong.myapp.UPDATE_APP_FILE_PROVIDER" android:exported="false" android:grantUriPermissions="true"> <!-- 元数据 --> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_provider_paths" /> </provider> ...... </application>
</manifest>

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

android:name:provider可以直接使用v4包提供的FileProvider,或者自定义,但是自定义的provider一定要继承FileProvider,一般使用系统的就足够了。
android:authorities:这个值将会发布在content://下,即其他app通过content://com.wong.myapp.UPDATE_APP_FILE_PROVIDER来访问apk文件等信息,稍后将在代码中用到。
**android:exported:“false”**表示我们的provider不需要对外开放。
android:grantUriPermissions:“true”,此值为true,app才能获得临时共享权限。
元数据meta-data中:
**android:name=“android.support.FILE_PROVIDER_PATHS”**必须是这个名字。因为在
android.support.v4.content.FileProvider类里,要用这个key,获取我们的xml文件路径。

XmlResourceParser in = info.loadXmlMetaData(context.getPackageManager(), "android.support.FILE_PROVIDER_PATHS");


  
 
  • 1
  • 2

android:resource="@xml/file_provider_paths"指定配置可访问路径的xml的文件地址。
第四步:res/xml中定义对外暴露的文件夹路径
首先,在res资源目录入创建xml文件夹,然后在xml创建一份名为file_provider_paths.xml的文件,内容如下:

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:android="http://schemas.android.com/apk/res/android"> <paths xmlns:android="http://schemas.android.com/apk/res/android"> <!--代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory()/DCIM/myapp目录--> <!--<external-path name="wong_DCIM" path="DCIM/myapp" />--> <!--代表外部存储区域的根目录下的文件 Environment.getExternalStorageDirectory()/Pictures/myapp目录--> <!--<external-path name="wong_Pictures" path="Pictures/myapp" />--> <!--代表app 外部存储区域根目录下的文件 Context.getExternalFilesDir(Environment.DIRECTORY_PICTURES)目录下的Pictures目录--> <!--/storage/emulated/0/Android/data/com.wong.myapp/files/Pictures--> <!--<external-files-path name="wong_external_files" path="Pictures" />--> <!--代表app 外部存储区域根目录下的文件 Context.getExternalCacheDir目录下的images目录--> <!--/storage/emulated/0/Android/data/com.wong.myapp/cache/images--> <!--<external-cache-path name="wong_external_cache" path="images" />--> <!--代表app 私有的存储区域 Context.getFilesDir()目录下的images目录 /data/user/0/com.wong.myapp/files/images--> <!--<files-path name="wong_private_files" path="images" />--> <!--代表app 私有的存储区域 Context.getCacheDir()目录下的images目录 /data/user/0/com.wong.myapp/cache/images--> <!--<cache-path name="wong_private_cache" path="images" />--> <root-path name="root_path" path="."/> </paths>

</resources>

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24

注意:
在android8.0以上的手机上,读取共享文件时,如apk更新的编程中,我们把apk下载后,要安装时,去读取这个apk文件就出现如下错误:

Failed to find configured root that contains
...

  
 
  • 1
  • 2

所以一定要在file_provider_paths.xml文件中添加root-path 标签,即:

<root-path name="root_path" path="."/>

  
 
  • 1

第五步:生成content://类型的Uri
使用以下代码只能生成格式为file://xxx的Uri,如果是android7.0之前的就用file://形式访问,获取Uri方式如下:

File picFile = xxx;
Uri picUri = Uri.fromFile(picFile);

  
 
  • 1
  • 2

android7.0以后的必须通过FileProvider.getUriForFile方法来生成content://xxx类型的Uri:


File apkFile = new File(Environment.getExternalStorageDirectory(), "update_folder/update.apk");
Uri fileUri = FileProvider.getUriForFile(MainActivity.this, "com.wong.myapp.UPDATE_APP_FILE_PROVIDER", apkFile);


  
 
  • 1
  • 2
  • 3
  • 4

getUriForFile:第一个参数是Context;第二个参数,就是我们之前在manifest#provider中定义的android:authorities属性的值;第三个参数是File

第六步:给Uri授予临时权限

intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

  
 
  • 1
  • 2

第七步:使用Intent传递Uri

intent.setDataAndType(fileUri, "application/vnd.android.package-archive");
startActivityForResult(intent, 100);

  
 
  • 1
  • 2

安装apk的核心代码:

 private void installApk() { Intent intent = new Intent(Intent.ACTION_VIEW); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.addCategory(Intent.CATEGORY_DEFAULT); //重新构造Uri:content:// Uri fileUri; File apkFile = new File(Environment.getExternalStorageDirectory(), "update_folder/update.apk"); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { fileUri = FileProvider.getUriForFile(MainActivity.this, "com.wong.myapp.UPDATE_APP_FILE_PROVIDER", apkFile); //授予目录临时共享权限 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); } else { fileUri = Uri.fromFile(apkFile); } intent.setDataAndType(fileUri, "application/vnd.android.package-archive"); //使用Intent传递Uri startActivity(intent); }

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

完整代码

文章来源: blog.csdn.net,作者:WongKyunban,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/weixin_40763897/article/details/88295946

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。