欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  技术分享

Android7.0调用相闪退_android7.0 fileuriexposedexception

程序员文章站 2022-02-02 09:43:41
...

最近又碰到因为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的留言板)

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