android 选择图片的权限问题

我们知道从android 6.0开始 系统开始对权限把控,7.0权限更是对系统权限进一步更改 主要就是三个方面

APP应用程序的私有文件不再向使用者放宽

Intent组件传递file://URI的方式可能给接收器留下无法访问的路径,触发FileUriExposedException异常,推荐使用FileProvider

DownloadManager不再按文件名分享私人存储的文件。旧版应用在访问COLUMN_LOCAL_FILENAME时可能出现无法访问的路径。面向 Android 7.0 或更高版本的应用在尝试访问 COLUMN_LOCAL_FILENAME 时会触发 SecurityException

一、深入理解FileProvider

FileProvider属于Android 7.0新增的一个类,该类位于v4包下,详情可见android.support.v4.content.FileProvider,使用方法类似与ContentProvider,简单概括为三个步骤,这里以调用sdcard公共目录安装app为例,演示使用过程:

在资源文件夹res/xml下新建file_provider.xml文件,文件声明权限请求的路径,代码如下:

   

       

   

在AndroidManifest.xml添加组件provider相关信息,类似组件activity,指定resource属性引用上一步创建的xml文件(后面会详细介绍各个属性的用法),代码如下:

    android:name="android.support.v4.content.FileProvider"

    android:authorities="com.wisdomclass.clus.fileprovider"

    android:grantUriPermissions="true"

    android:exported="false">

   

        android:name="android.support.FILE_PROVIDER_PATHS"

        android:resource="@xml/provider_paths" />

最后一步,Java代码申请权限,使用新增的方法getUriForFile()和grantUriPermission(),代码如下(后面会详细介绍方法对应参数的使用):

public void InstallApk(){

    String filePath = getExternalFilesDir("Download").getAbsolutePath() + File.separator+"软件名称"+mVersion+".apk";

    Intent install = new Intent(Intent.ACTION_VIEW);

    if(Build.VERSION.SDK_INT>=24) {//判读版本是否在7.0以上

        Uri apkUri = FileProvider.getUriForFile(this, BuildConfig.APPLICATION_ID +".fileprovider", new File(filePath));

        install.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

        install.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);//添加这一句表示对目标应用临时授权该Uri所代表的文件

        install.setDataAndType(apkUri, "application/vnd.android.package-archive");

    }else {

        install.setDataAndType(Uri.parse("file://"+filePath), "application/vnd.android.package-archive");

        install.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

    }

    startActivity(install);

}

1.1 定义一个FileProvider

直接使用FileProvider本身或者它的子类,需要在AndroidManifest.xml文件中声明组件的相关属性,包括:

android:name,对应属性值:android.support.v4.content.FileProvider或者子类完整路径

android:authorities,对应属性值是一个常量,通常定义的方式packagename.fileprovider,例如:cn.teachcourse.fileprovider

android:exported,对应属性值是一个boolean变量,设置为false

android:grantUriPermissions,对应属性值也是一个boolean变量,设置为true,允许获得文件临时的访问权限

    ...

   

        ...

       

            android:name="android.support.v4.content.FileProvider"

            android:authorities="com.mydomain.fileprovider"

            android:exported="false"

            android:grantUriPermissions="true">

            ...

       

        ...

   


想要关联res/xml文件夹下创建的file_provider.xml文件,需要在标签内,添加子标签,设置标签的属性值,包括:

android:name,对应属性值是一个固定的系统常量android.support.FILE_PROVIDER_PATHS

android:resource,对应属性值指向我们的xml文件@xml/file_provider

    android:name="android.support.v4.content.FileProvider"

    android:authorities="com.mydomain.fileprovider"

    android:exported="false"

    android:grantUriPermissions="true">

   

        android:name="android.support.FILE_PROVIDER_PATHS"

        android:resource="@xml/file_provider" />

1.2 指定授予临时访问权限的文件目录

上一步说明了怎么定义一个FileProvider,这一步主要说明怎么定义一个@xml/file_provider文件。Android Studio或Eclipse开发工具创建Android项目的时候默认不会创建res/xml文件夹,需要开发者手动创建,点击res文件夹新建目录,命名xml,如下图:

然后,在xml文件夹下新建一个xml文件,文件命名file_provider.xml,指定根标签为paths,如下图:

在xml文件中指定文件存储的区块和区块的相对路径,在根标签中添加子标签(稍后详细列出所有子标签),设置子标签的属性值,包括:

name,是一个虚设的文件名(可以自由命名),对外可见路径的一部分,隐藏真实文件目录

path,是一个相对目录,相对于当前的子标签根目录

,表示内部内存卡根目录,对应根目录等价于Context.getFilesDir(),查看完整路径:

/data/user/0/cn.teachcourse.demos/files

代码如下:

   

    ...

根标签下可以添加的子标签也是有限的,参考官网的开发文档,除了上述的提到的这个子标签外,还包括下面几个:

,表示应用默认缓存根目录,对应根目录等价于getCacheDir(),查看完整路径:/data/user/0/cn.teachcourse.demos/cache

,表示外部内存卡根目录,对应根目录等价于

Environment.getExternalStorageDirectory(),

查看完整路径:/storage/emulated/0

,表示外部内存卡根目录下的APP公共目录,对应根目录等价于

Context#getExternalFilesDir(String) Context.getExternalFilesDir(null),

查看完整路径:

/storage/emulated/0/Android/data/cn.teachcourse.demos/files/Download

,表示外部内存卡根目录下的APP缓存目录,对应根目录等价于Context.getExternalCacheDir(),查看完整路径:

/storage/emulated/0/Android/data/cn.teachcourse.demos/cache

你可能感兴趣的:(android 选择图片的权限问题)