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" />
第二步:动态申请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>
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");
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>
注意:
在android8.0以上的手机上,读取共享文件时,如apk更新的编程中,我们把apk下载后,要安装时,去读取这个apk文件就出现如下错误:
Failed to find configured root that contains
...
所以一定要在file_provider_paths.xml文件中添加root-path 标签,即:
<root-path name="root_path" path="."/>
第五步:生成content://类型的Uri
使用以下代码只能生成格式为file://xxx的Uri,如果是android7.0之前的就用file://形式访问,获取Uri方式如下:
File picFile = xxx;
Uri picUri = Uri.fromFile(picFile);
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);
getUriForFile:第一个参数是Context;第二个参数,就是我们之前在manifest#provider中定义的android:authorities属性的值;第三个参数是File。
第六步:给Uri授予临时权限
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION
| Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
第七步:使用Intent传递Uri
intent.setDataAndType(fileUri, "application/vnd.android.package-archive");
startActivityForResult(intent, 100);
安装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);
}