解决Android10版本以上外部存储权限的方案

Posted 一盏清风予明月

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了解决Android10版本以上外部存储权限的方案相关的知识,希望对你有一定的参考价值。

项目场景:主要解决android10版本以上的分区机制(socoped storage)

分区存储机制,又叫做沙盒存储机制,用于防止应用读取其他应用的数据。每个应用程序都应该有自己的存储空间。应用程序不能翻过自己的目录,去访问公共目录。因此,用用程序在请求数据时要通过权限检测,不符合要求的不会被放行。


问题描述

我在学习android过程中,由于我安装的是android11的版本,一次实践VideoView播放SD卡中存放的视频中发现了它不能简单地通过在AndroidManifest.xml文件中添加

<uses—permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>

来实现对SD卡的外部存储实现数据读取。在我查找了许多博主的博客,发现好多都是老版本的方案,初学者表示很懵逼。最终功夫不负有心人,我最终在几个博主的博客里找到正确的解决方案。现把我的解决方案整理一下,供大家参考。


原因分析:

在发布android10的时候官方明确表态:

2020年,主要平台版本将要求所有应用都使用分区存储,无论应用的目标 sdk 级别是多少。因此,您应该提前确保您的应用能够使用分区存储。为此,请确保针对搭载 android 10(api 级别 29)及更高版本的设备启用了该行为。

以 android 10(api 级别 29)及更高版本为目标平台的应用在默认情况下被赋予了对外部存储设备的分区访问权限(即分区存储), 对外部存储文件访问方式重新设计,便于用户更好的管理外部存储文件。如果不符合条件的会以兼容模式运行,兼容模式跟以前一样,根据路径可以直接存储文件。

应用只能看到本应用专有的目录(通过 context.getexternalfilesdir() 访问)以及特定类型的媒体。除非您的应用需要访问存放在应用的专有目录以及 mediastore 之外的文件,否则最好使用分区存储。

具体分区存储权限的介绍

默认情况下,对于targetsdkversion大于等于29的应用,其访问权限范围限定为分区存储。此应用无需请求与存储相关的用户权限,即可以查看外部存储中以下类型的文件:

应用外部特定目录中的文件(使用getexternalfilesdir()访问)。

应用自己创建的照片、视频和音频(通过mediastore访问)。

分区存储将影响在android10系统首次安装启动、且targetsdkversion >=29的应用。需要访问和共享外部存储文件的应用会受到影响,需要进行兼容性适配。

影响范围:

在android 10上运行的应用:

1.targetsdkversion <= 28,不受影响

2.如果targetsdkversion >= 29,默认情况应用外部存储可见性将被过滤,应用需要对分区存储进行适配。

还有值得注意的是以下两种情况比较特殊,不会受到分区存储的影响:

如果应用最先安装在android 10以下的系统,

1) 然后系统通过fota升级到android 10

2) 应用通过更新升级到targetsdkversion >= 29


解决方案:

方法1:

通过安卓的第三方框架导入依赖:

【Android 中如何导入依赖包】https://mbd.baidu.com/ma/s/vdFk0yAe

buildscript 
        repositories 
            maven  url 'https://jitpack.io' 
        
    
    allprojects 
        repositories 
            maven  url 'https://jitpack.io' 
        
    

dependencies

// 权限请求框架:https://github.com/getActivity/XXPermissions
implementation 'com.github.getActivity:XXPermissions:13.2'

 

XXPermissions.with(this)
        // 不适配 Android 11 可以这样写
        //.permission(Permission.Group.STORAGE)
        // 适配 Android 11 需要这样写,这里无需再写 Permission.Group.STORAGE
        .permission(Manifest.permission.MANAGE_EXTERNAL_STORAGE)
        .request(new OnPermissionCallback() 

            @Override
            public void onGranted(List<String> permissions, boolean all) 
                if (all) 
                    toast("获取存储权限成功");
                
            

            @Override
            public void onDenied(List<String> permissions, boolean never) 
                if (never) 
                    toast("被永久拒绝授权,请手动授予存储权限");
                    // 如果是被永久拒绝就跳转到应用权限系统设置页面
                    XXPermissions.startPermissionActivity(MainActivity.this, permissions);
                 else 
                    toast("获取存储权限失败");
                
            
        );

 然后在进行对外部存储如SD卡的访问,就当笔者以为终于大功告成时,但在在进行调试时又报了个错误android10打开文件异常 open failed: EACCES (Permission denied) android:requestLegacyExternalStorage=“true“,查阅资料后,在AndroidManifest.xml文件中application节点中加上android:requestLegacyExternalStorage="true"属性就可以了,如下:

<application
    android:requestLegacyExternalStorage="true"

运行项目,即可访问到SD卡中/storage/emulated/0/video.mp4"(或/mnt/sdcard/video.mp4)中的视频。

方法2:申请外部存储权限适配

Android 6.0 以上——10.0以下的外部存储权限适配

Android 10.0 以下外部存储权限适配

  • 升级 targetSdkVersion
    android 
        defaultConfig 
            targetSdkVersion 23
        
    

  • 添加清单权限
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

  • 代码动态申请
    public final class PermissionActivity extends AppCompatActivity 
    
        private static final int REQUEST_CODE = 1024;
    
        @Override
        protected void onCreate(@Nullable Bundle savedInstanceState) 
            super.onCreate(savedInstanceState);
            requestPermission();
        
    
        private void requestPermission() 
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) 
                // 先判断有没有权限
                if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
                        ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) 
                    writeFile();
                 else 
                    ActivityCompat.requestPermissions(this, new String[]Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE, REQUEST_CODE);
                
             else 
                writeFile();
            
        
    
        @Override
        public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) 
            super.onRequestPermissionsResult(requestCode, permissions, grantResults);
            if (requestCode == REQUEST_CODE) 
                if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
                        ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) 
                    writeFile();
                 else 
                    ToastUtils.show("存储权限获取失败");
                
            
        
    
        /**
         * 模拟文件写入
         */
        private void writeFile() 
            ToastUtils.show("写入文件成功");
        
    

  • 需要注意的是,如果 targetSdkVersion >= 29 上,还需要在清单文件中加上
    <application
        android:requestLegacyExternalStorage="true">

    否则就算申请了存储权限,在安卓 10.0 的设备上将无法正常读写外部存储上的文件

  • Android 11 及以上申请外部存储权限

  • 升级 targetSdkVersion
  • android 
        defaultConfig 
            targetSdkVersion 30
        
    

 添加清单权限

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" />

 

  • 代码动态申请
public final class PermissionActivity extends AppCompatActivity 

    private static final int REQUEST_CODE = 1024;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        requestPermission();
    

    private void requestPermission() 
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) 
            // 先判断有没有权限
            if (Environment.isExternalStorageManager()) 
                writeFile();
             else 
                Intent intent = new Intent(Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSION);
                intent.setData(Uri.parse("package:" + context.getPackageName()));
                startActivityForResult(intent, REQUEST_CODE);
            
         else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) 
            // 先判断有没有权限
            if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
                    ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) 
                writeFile();
             else 
                ActivityCompat.requestPermissions(this, new String[]Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE, REQUEST_CODE);
            
         else 
            writeFile();
        
    

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) 
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == REQUEST_CODE) 
            if (ActivityCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED &&
                    ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) 
                writeFile();
             else 
                ToastUtils.show("存储权限获取失败");
            
        
    

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) 
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == REQUEST_CODE && Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) 
            if (Environment.isExternalStorageManager()) 
                writeFile();
             else 
                ToastUtils.show("存储权限获取失败");
            
        
    

    /**
     * 模拟文件写入
     */
    private void writeFile() 
        ToastUtils.show("写入文件成功");
    

调试项目后可以正常访问到外部存储中如SD卡中的视频文件了。

附上我的源码供大家参考:http://链接:https://pan.baidu.com/s/1AWlUsL0oZlZxzghqbAt0_w

 提取码:78c0

Android 29及以上高版本获取SD外部存储器目录

Android 29及以上高版本获取SD外部存储器目录/storage/emulated/0

     // 获得SD外部存储根目录一般为:/storage/emulated/0
    public static File getSDRootPath(Context context) 
        File sdPath;
        boolean isSDExist = Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED); //SD卡是否存在
        if (isSDExist) 
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) 
                File externalFileRootDir = context.getExternalFilesDir("");
                do 
                    externalFileRootDir = Objects.requireNonNull(externalFileRootDir).getParentFile();
                 while (Objects.requireNonNull(externalFileRootDir).getAbsolutePath().contains("/Android"));
                sdPath = Objects.requireNonNull(externalFileRootDir);
             else 
                sdPath = context.getExternalFilesDir("");
            
         else 
            sdPath = Environment.getRootDirectory();//根目录
        

        return sdPath;
    

以上是关于解决Android10版本以上外部存储权限的方案的主要内容,如果未能解决你的问题,请参考以下文章

Android——Android10的分区存储(Scoped Storage)

Android 29及以上高版本获取SD外部存储器目录

Android 29及以上高版本获取SD外部存储器目录

解决Android无法访问与下载文件的权限问题

以编程方式测试是否在Android 10+中禁用了旧版外部存储访问

Unity在Android 6.0及以上版本弹出权限申请窗口的问题