美文网首页Android
Android 7.0权限适配:FileUriExposedEx

Android 7.0权限适配:FileUriExposedEx

作者: joker_fu | 来源:发表于2017-07-25 22:14 被阅读119次

今天来聊聊Android 7.0 FileUriExposedException异常,以及它的使用方法和使用场景

一 描述

  1. 问题
    对于面向 Android 7.0 的应用,Android 框架执行的 StrictModeAPI 政策禁止在您的应用外部公开 file:// URI。如果一项包含文件 URI 的 intent 离开您的应用,则应用出现故障,并出现 FileUriExposedException异常
  2. 解决方案
    要在应用间共享文件,您应发送一项 content://URI,并授予 URI 临时访问权限。进行此授权的最简单方式除了将targetSdkVersion改成24以下,就是使用 FileProvider

官网对FileProvider描述:

FileProvider是ContentProvider的一个特殊子类,它通过创建内容来实现与应用程序相关联的文件的安全共享:// Uri用于文件,而不是文件:/// Uri。

内容URI允许您使用临时访问权限来授予读取和写入访问权限。当您创建包含内容URI的Intent时,为了将内容URI发送到客户端应用程序,还可以调用Intent.setFlags()来添加权限。只要接收活动的堆栈处于活动状态,客户端应用程序就可以使用这些权限。对于要访问服务的意图,只要服务正在运行,权限就可用。

相比之下,为了控制对文件的访问:/// Uri你必须修改底层文件的文件系统权限。您提供的权限可用于任何应用程序,并在您更改之前保持有效。这种访问水平基本上是不安全的。

内容URI提供的增加文件访问安全级别使FileProvider成为Android安全基础架构的关键部分。

二 如何使用FileProvider

我们先看如何使用FileProvider,官网也有详细说明:https://developer.android.com/reference/android/support/v4/content/FileProvider.html

1. 定义FileProvider

由于FileProvider的默认功能,包括内容URI代的文件,你不需要在代码中定义一个子类。我们在manifest中声明provider

<manifest>
    ...
    <application>
        ...
        <provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="包名.fileprovider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/file_paths"/>
        </provider>
        ...
    </application>
</manifest>

android:name 【固定值】 FileProvider的包名+类名
android:authorities 【自定义】 推荐以包名+”.fileprovider”方式命名,增加辨别性,系统唯一
android:exproted 要求必须为false,为true则会报安全异常
android:grantUriPermissions 是否允许为文件设置临时权限 “true”
android:resource="@xml/file_paths"就是我们的共享路径配置的xml文件

2 . 配置file_paths

FileProvider只能生成你事先指定的 content URI,file_paths配置如下:

<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path
        name="external"
        path=""/>
    <external-path
        name="my_images"
        path="Android/data/包名/files/Pictures/"/>
    <external-path
        name="images"
        path="Pictures/"/>
</paths>

<small>注意: 注: XML文件是你可以指定你要共享的目录的唯一途径,你不能以编程方式添加一个目录,至少配置一个external-path节点</small>

在paths节点内部支持以下几个子节点,分别为:

  • <root-path/> 代表设备的根目录new File("/")

  • <files-path/> 代表该文件files/的应用程序的内部存储区的子目录,等同于context.getFilesDir()

  • <cache-path/> 代表应用程序的内部存储区域的缓存子目录的文件,等同于context.getCacheDir()

  • <external-path/> 代表在外部存储区根目录的文件,等同于Environment.getExternalStorageDirectory()

  • <external-files-path> 代表应用程序的外部存储区根目录的文件,等同于Context.getExternalFilesDir(String) /Context.getExternalFilesDir(null)

  • <external-cache-path> 代表应用程序的外部缓存区根目录的文件,等同于Context.getExternalCacheDir()

file_paths用来指定Uri共享和真实路径的映射关系,name属性的值可以自定义,path属性的值表示共享的具体位置,设置为空,就表示共享整个SD卡,也可指定对应的SDcard下的文件目录,根据需求自行定义

3. 获得content uri

使用getUriForFile()将file:// 转换成 content://
Uri fileUri = FileProvider.getUriForFile(this, "包名.fileprovider", file);

4. 临时读写权限授权

需要对接收应用设置读权限或写权限亦或读写均设置:
FLAG_GRANT_READ_URI_PERMISSION:读权限
FLAG_GRANT_WRITE_URI_PERMISSION:写权限
授权方式:

  1. 使用Intent.addFlags或setFlags,该方式授权的有效期限,权限截止于该 App 所处的堆栈被销毁自动回收(APP销毁),主要用于针对intent.setData,setDataAndType以及setClipData相关方式传递uri
    2 使用grantUriPermission(String toPackage, Uri uri, int modeFlags)来进行授权,该方式授权的有效期限,从授权一刻开始,手动调用 Context.revokeUriPermission() 方法或者设备重启才截止

三 使用场景

a. 相机拍照
Android 7.0之前我们这样拍照,没有什么问题(忽略6.0权限问题):

    private static final int REQUEST_TAKE_PHOTO = 0X11;
    private Uri imageUri ;
       private void takePhoto() {
        Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        //判断是否有相机应用
        if (takePictureIntent.resolveActivity(getActivity().getPackageManager()) != null) {
            //获取存储路径 没有则创建
            File directory = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
            if (!directory.exists()) {
                if (!directory.mkdir()) {
                    return;
                }
            }
            File file = new File(directory.getAbsolutePath(), new SimpleDateFormat("yyyyMMdd-HHmmss", Locale.CHINA)
                    .format(new Date()) + ".jpeg");
            imageUri = Uri.fromFile(file);
            takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
            startActivityForResult(takePictureIntent, TAKE_PHOTO);
        } else {
            ToastUtil.showShort(getString(R.string.TakePhoto_Error));
        }
    }

    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (resultCode == RESULT_OK && requestCode == REQUEST_TAKE_PHOTO) {
            // 通知图库更新
            getActivity().sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, imageUri ));

        }
    }

如果我们使用Android 7.0或者以上的原生系统运行,发现应用直接停止运行,如文章开头所说抛出了android.os.FileUriExposedException:

android.os.FileUriExposedException: 
    file:///storage/emulated/0/Pictures/20170723-201847.jpeg exposed   beyond app through ClipData.Item.getUri()
    at android.os.StrictMode.onFileUriExposed(StrictMode.java:1799)
    at android.net.Uri.checkFileUriExposed(Uri.java:2346)

接下来根据官网的解决办法,如第二步所说配置好 FileProvider,更改拍照方法:

private void takePhoto() {
        Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        //判断是否有相机应用
        if (takePictureIntent.resolveActivity(getActivity().getPackageManager()) != null) {
            //获取存储路径 没有则创建
            File directory = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
            if (!directory.exists()) {
                if (!directory.mkdir()) {
                    return;
                }
            }
            File file = new File(directory.getAbsolutePath(), new SimpleDateFormat("yyyyMMdd-HHmmss", Locale.CHINA)
                    .format(new Date()) + ".jpeg");
            Uri uri = imageUri = Uri.fromFile(file);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                //兼容7.0
                uri = FileProvider.getUriForFile(getApplication(), "包名.fileprovider", file);
                //添加权限 这一句表示对目标应用临时授权该Uri所代表的文件
                takePictureIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                takePictureIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            }
            takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
            startActivityForResult(takePictureIntent, TAKE_PHOTO);
        } else {
            ToastUtil.showShort(getString(R.string.TakePhoto_Error));
        }
    }

添加了版本判断,并使用 FileProvider.getUriForFile()获得content Uri,方法主要更改如下:

     if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        //兼容7.0
        uri = FileProvider.getUriForFile(getApplication(), "包名.fileprovider", file);
        //添加权限 这一句表示对目标应用临时授权该Uri所代表的文件
        takePictureIntent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        takePictureIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
    }

当然也可以不用判断版本,直接使用FileProvider.getUriForFile(getApplication(), "包名.fileprovider", file)获得Uri替换Uri.fromFile(file),但是切记需要进行授权和取消授权,否则4.4以下会报Permission Denial

b. 图片裁剪

/**
 * @param activity    当前activity
 * @param orgUri      剪裁原图的Uri
 * @param desUri      剪裁后的图片的Uri
 * @param aspectX     X方向的比例
 * @param aspectY     Y方向的比例
 * @param width       剪裁图片的宽度
 * @param height      剪裁图片高度
 * @param requestCode 剪裁图片的请求码
 */
public static void cropImageUri(Activity activity, Uri orgUri, Uri desUri, int aspectX, int aspectY, int width, int height, int requestCode) {
        Intent intent = new Intent("com.android.camera.action.CROP");
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        }
        intent.setDataAndType(orgUri, "image/*");
        intent.putExtra("crop", "true");
        intent.putExtra("aspectX", aspectX);
        intent.putExtra("aspectY", aspectY);
        intent.putExtra("outputX", width);
        intent.putExtra("outputY", height);
        intent.putExtra("scale", true);
        //将剪切的图片保存到目标Uri中
        intent.putExtra(MediaStore.EXTRA_OUTPUT, desUri);
        intent.putExtra("return-data", false);
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
        intent.putExtra("noFaceDetection", true);
        activity.startActivityForResult(intent, requestCode);
    }

c. 安装apk

// 安装Apk
public void installApk(Context context) {
    File file = new File(Environment.getExternalStorageDirectory(), "app.apk");
    Intent intent = new Intent(Intent.ACTION_VIEW);
    Uri uri = Uri.fromFile(file);
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        uri = FileProvider.getUriForFile(context, "包名.fileprovider", file);
        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
    }
    intent.setDataAndType(uri, "application/vnd.android.package-archive");
    context.startActivity(intent);
}

大概使用就这么多,望多多指教。

另附上:官网学习使用FileProvider地址

相关文章

网友评论

    本文标题:Android 7.0权限适配:FileUriExposedEx

    本文链接:https://www.haomeiwen.com/subject/bnsmkxtx.html