Android7.0适配总结

Android7.0适配注意事项

权限更改

  • Android6.0引入了动态权限控制(Runtime Permission),Android7.0升级到“私有目录被限制访问”即“StrictMode API 政策”。这些更改为用户带来了更加安全的操作系统。

权限更改带来的影响

目录被限制访问

  • 私有文件的文件权限不在放权给所有的应用,使用 MODE_WORLD_READABLEMODE_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;
    }
}

 

参考

Android N系列适配---FileProvider

Android7.0适配教程,心得

Android 7.0 开发者版本

官方文档FileProvide

推荐阅读更多精彩内容