Android7.0适配注意事项
权限更改
- Android6.0引入了动态权限控制(Runtime Permission),Android7.0升级到“私有目录被限制访问”即“StrictMode API 政策”。这些更改为用户带来了更加安全的操作系统。
权限更改带来的影响
目录被限制访问
- 私有文件的文件权限不在放权给所有的应用,使用
MODE_WORLD_READABLE
或MODE_WORLD_WRITEABLE
进行的操作将触发 SecurityException。
应对策略:这项权限的变更将意味着你无法通过File API访问手机存储上的数据了,基于File API的一些文件浏览器等也将受到很大的影响,看到这大家是不是惊呆了呢,不过迄今为止,这种限制尚不能完全执行。 应用仍可能使用原生 API 或 File API 来修改它们的私有目录权限。 但是,Android官方强烈反对放宽私有目录的权限。可以看出收起对私有文件的访问权限是Android将来发展的趋势。
- 给其他应用传递 file:// URI 类型的Uri,可能会导致接受者无法访问该路径。 因此,在Android7.0中尝试传递 file:// URI 会触发 FileUriExposedException。
应对策略:大家可以通过使用FileProvider来解决这一问题。
- DownloadManager 不再按文件名分享私人存储的文件。COLUMN_LOCAL_FILENAME在Android7.0中被标记为deprecated ,旧版应用在访问 COLUMN_LOCAL_FILENAME时可能出现无法访问的路径。 面向 Android N 或更高版本的应用在尝试访问 COLUMN_LOCAL_FILENAME 时会触发 SecurityException。
应对策略:大家可以通过ContentResolver.openFileDescriptor()来访问由 DownloadManager 公开的文件。
应用间共享文件
在Android7.0系统上,Android 框架强制执行了 StrictMode API 政策禁止向你的应用外公开 file:// URI。 如果一项包含文件 file:// URI类型 的 Intent 离开你的应用,应用失败,并出现 FileUriExposedException 异常,如调用系统相机拍照,或裁切照片。
应对策略:若要在应用间共享文件,可以发送 content:// URI类型的Uri,并授予 URI 临时访问权限。 进行此授权的最简单方式是使用 FileProvider类。 如需有关权限和共享文件的更多信息,请参阅共享文件。
解决方案
-
第一步:
- 全局找出项目中,需要修改的地方,如下:
- Uri.parse、Uri.fromFile、file://、content://、Context.getFilesDir()、Environment.getExternalStorageDirectory()、getCacheDir()以及最终要的intent.setDataAndType(为什么需要找这个,因为这个会携带uri进行传递,这个是重头戏)
-
第二步:
- 找到罪魁祸首之后,需要按照步骤适配了,依次顺序是,AndroidManifest.xml清单文件的修改,资源文件的修改,以及Java代码中的修改
-
第三步:
<!-- 适配Android7.0 FileProvider-->
<manifest>
...
<application>
...
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="package_name.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
</application>
</manifest>
心得:exported:要求必须为false,为true则会报安全异常。grantUriPermissions:true,表示授予 URI 临时访问权限。
- 第四步:
- 为了指定共享的目录我们需要在资源(res)目录下创建一个xml目录,创建res/xml/filepaths.xml文件
<?xml version="1.0" encoding="utf-8"?>
<resources>
<paths>
<external-path
name="external-path"
path="" />
<files-path
name="files_path"
path="" />
<cache-path
name="cache-path"
path="" />
<root-path
name="root-path"
path="" />
</paths>
</resources>
- <files-path/>代表的根目录: Context.getFilesDir()
- <external-path/>代表的根目录: Environment.getExternalStorageDirectory()
- <cache-path/>代表的根目录: getCacheDir()
- < root-path/>国内由于rom众多,会产生各种路径,比如华为的/system/media/,以及外置sdcard,
-
第五步:
- 在Java代码中使用:
//得到缓存路径的Uri
Uri contentUri = FileProvider.getUriForFile(getActivity(), "com.***.fileprovider", file);
//获取壁纸
Intent intent = WallpaperManager.getInstance(getActivity()).getCropAndSetWallpaperIntent(contentUri);
//开启一个Activity显示图片,可以将图片设置为壁纸。调用的是系统的壁纸管理。
getActivity().startActivityForResult(intent, ViewerActivity.REQUEST_CODE_SET_WALLPAPER);
- 需要的权限,intent携带的读写权限
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
- 在适配过程中,发现有时候addFlag并不能完全的拥有权限,需要grantUriPermission获取权限
context.grantUriPermission(packageName, uri,
Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
工具类:
public class NougatTools {
private static final String Nougat_FileProvider = "cn.yupaopao.common.fileprovider";
/**
* 将普通uri转化成适应7.0的content://形式 针对文件格式
*
* @param context 上下文
* @param file 文件路径
* @param intent intent
* @param intentType intent.setDataAndType
* @return Intent
*/
public static Intent formatFileProviderIntent(Context context, File file, Intent intent, String intentType) {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
Uri uri = Uri.fromFile(file);
intent.setDataAndType(uri, intentType);
} else {
Uri uri = FileProvider.getUriForFile(context, Nougat_FileProvider, file);
// 表示文件类型
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.setDataAndType(uri, intentType);
}
return intent;
}
/**
* 将普通uri转化成适应7.0的content://形式 针对图片格式
*
* @param context 上下文
* @param file 文件路径
* @param intent intent
* @return Intent
*/
public static Intent formatFileProviderPicIntent(Context context, File file, Intent intent) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
Uri uri = FileProvider.getUriForFile(context, Nougat_FileProvider, file);
List<ResolveInfo> resInfoList = context.getPackageManager().queryIntentActivities(
intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resInfoList) {
String packageName = resolveInfo.activityInfo.packageName;
context.grantUriPermission(packageName, uri,
Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
// 表示图片类型
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
}
return intent;
}
/**
* 将普通uri转化成适应7.0的content://形式
*
* @return Uri
*/
public static Uri formatFileProviderUri(Context context, File file) {
Uri uri;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
uri = Uri.fromFile(file);
} else {
uri = FileProvider.getUriForFile(context, Nougat_FileProvider, file);
}
return uri;
}
}