Android7.0调用相闪退_android7.0 fileuriexposedexception
2017-10-24 13:48:29  By: shinyuu

最近又碰到因为android 7.0 引起的兼容问题了。之前一次是版本跟新安装APK的时候,这次是调用相机拍照的时候。有遇到apk安装问题的哥们可以去看看 Android7.0 更新APK报错适配 ,这次是拍照的问题,解决方案类似。


在7.0以前的版本:

这个file文件直接非常简单的转换成"file://XXX/XXX/XXX"的uri格式

private void startCamera() {
    Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    File tmpFile = new File(CommonConstants.GOODCARFORSELLER_IMAGES, CAMERA_FILE_NAME);
    Uri outputFileUri = Uri.fromFile(tmpFile);
    intent.putExtra(MediaStore.EXTRA_OUTPUT, outputFileUri);
    startActivityForResult(intent, REQUEST_CAMERA);
}


7.0后的版本:

当把targetSdkVersion指定成24及之上并且在API>=24的设备上运行时。这种方式则会出现FileUriExposedException异常

android.os.FileUriExposedException:file:///XXX exposed beyond app through ClipData.Item.getUri()
at android.os.StrictMode.onFileUriExposed(StrictMode.java:1799)
at android.net.Uri.checkFileUriExposed(Uri.java:2346)
at android.content.ClipData.prepareToLeaveProcess(ClipData.java:832)
at android.content.Intent.prepareToLeaveProcess(Intent.java:8909)
...



原因

Android不再允许在app中把file://Uri暴露给其他app,包括但不局限于通过Intent或ClipData 等方法。

原因在于使用file://Uri会有一些风险,比如:

文件是私有的,接收file://Uri的app无法访问该文件。在Android6.0之后引入运行时权限,如果接收file://Uri的app没有申请READ_EXTERNAL_STORAGE权限,在读取文件时会引发崩溃。


因此,google提供了FileProvider,使用它可以生成content://Uri来替代file://Uri。


解决方案

1、AndroidManifest.xml中添加provider

android:authorities 是用来标识provider的唯一标识,在同一部手机上一个"authority"串只能被一个app使用,冲突的话会导致app无法安装。

android:exported 必须设置成false,后面异常会讲为什么。

android:grantUriPermissions 用来控制共享文件的访问权限,也可以在java代码中设置。

<application
    android:name=".GoodCarForSellerApplication"
    android:allowBackup="true"
    android:hardwareAccelerated="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:largeHeap="true"
    android:roundIcon="@mipmap/ic_launcher_round"
    android:supportsRtl="true"
    android:theme="@style/AppTheme.NoActionBar">

    <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>


2、res/xml/provider_paths.xml

files-path 节点路径 Context.getFilesDir()

cache-path 节点路径 Context.getCacheDir()

external-path 节点路径 Environment.getExternalStorageDirectory()

external-files-path 节点路径 Context.getExternalFilesDir(null)

external-cache-path 节点路径 Context.getExternalCacheDir()

<?xml version="1.0" encoding="utf-8"?>
<paths>
    <external-path path="Android/data/你的包名/" name="files_root" />
    <external-path path="." name="external_storage_root" />
</paths>


3、然后修改代码

private void startCamera() {
    Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    File tmpFile = new File(CommonConstants.GOODCARFORSELLER_IMAGES, CAMERA_FILE_NAME);
    //Uri outputFileUri = Uri.fromFile(tmpFile);
    Uri outputFileUri = FileProvider.getUriForFile(this, BuildConfig.APPLICATION_ID   ".fileProvider", tmpFile);
    intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
    intent.putExtra(MediaStore.EXTRA_OUTPUT, outputFileUri);
    startActivityForResult(intent, REQUEST_CAMERA);
}


异常处理

1、java.lang.SecurityException: Provider must not be exported

解决方案:android:exported必须设置成false


2、Attempt to invoke virtual method ´android.content.res.XmlResourceParser android.content.pm.PackageItemInfo.loadXmlMetaData(android.content.pm.PackageManager, java.lang.String)´ on a null object reference

解决方案:AndroidManifest.xml处的android:authorities 必须跟 BuildConfig.APPLICATION_ID ".fileprovider"一样


若资源对你有帮助,浏览后有很大收获,不妨小额打赏我一下,你的鼓励是维持我不断写博客最大动力。

想获取DD博客最新代码,你可以扫描下方的二维码,关注DD博客微信公众号(ddblogs)。

或者你也可以关注我的新浪微博,了解DD博客的最新动态:DD博客官方微博(dwtedx的微博)

如对资源有任何疑问或觉得仍然有很大的改善空间,可以对该博文进行评论,希望不吝赐教。

为保证及时回复,可以使用博客留言板给我留言: DD博客留言板(dwtedx的留言板)

感谢你的访问,祝你生活愉快、工作顺心,欢迎常来逛逛。


快速评论


技术评论

  • 该技术还没有评论、赶快抢沙发吧...
DD记账
top
+