以下均用kotlin编写,没用过的请酌情参考,方法类似

1、首先Manifest中添加权限

这是Manifest文件的开头

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools">
<!--    必须要加下面两句-->
    <uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

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

这两句必须加,但是上面那句会提示这样不能在谷歌商店上架,我自己编软件自己用所以不担心,如果真要商用,建议还是老老实实存私有空间,Android11开始应用好像不能在公共空间操作了,需要用户手动设置。

2、动态获取权限

主Activity的class里定义

private val readRequest = android.Manifest.permission.READ_EXTERNAL_STORAGE

在主Activity中的onCreate的方法中

val readFlag = ActivityCompat.checkSelfPermission(this,readRequest) != PackageManager.PERMISSION_GRANTED
//        var writeFlag = ActivityCompat.checkSelfPermission(this, writeContacts) != PackageManager.PERMISSION_GRANTED
            if(readFlag){
                if(ActivityCompat.shouldShowRequestPermissionRationale(this,readRequest)){
                    //引导用户打开读取联系人权限
                }

                ActivityCompat.requestPermissions(this,
                    arrayOf(readRequest),
                    100
                )

            }else{
                //getContacts()
            }

这个方法好像就是点开应用提示获取权限,当然这个权限是只能访问media文件(我不清楚这是什么类型的文件,就类似MP4格式?)

然后添加获取权限结果的函数,在onCreate方法外

当然我这里什么都没干,就弹了个提醒而已

override fun onRequestPermissionsResult(
        requestCode: Int,
        permissions: Array<out String>,
        grantResults: IntArray
    ) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults)

        if (requestCode == 100) {
            if (grantResults.contains(PackageManager.PERMISSION_GRANTED)) {
                if (permissions.contains(readRequest)) {
                    Toast.makeText(this,"成功",Toast.LENGTH_SHORT).show()
                }

            }
        }
    }

好的走到这里已经做完了网上搜到的90%的教程了,当然剩下还有什么在Manifest的<application>标签中添加

android:requestLegacyExternalStorage=“true”

android:preserveLegacyExternalStorage=“true”

之类的都在Android11不管用了,低于11的可以试试。

3、接下来,正常运行代码,打开应用后进行授权

 选择同意,之后切到后台,去系统设置进行手动用户授权

 之后切回app,就能正常使用外部公共空间了!

-----------------------------------------------------------------------------------------------------------------------

如果你看到了这里,那么你应该去看一下评论(不知道后面会不会没),评论大佬提到了一种Scoped Storage,去搜了一下发现有大佬详细解释了这种特性,

附上链接Android 11新特性,Scoped Storage又有了新花样_"tools:ignore=\"scopedstorage"_guolin的博客-CSDN博客

里面提到了获取权限的简单方法,不需要再像上面那样手动开启权限了,代码如下

if (Build.VERSION.SDK_INT < Build.VERSION_CODES.R ||
        Environment.isExternalStorageManager()) {
    Toast.makeText(this, "已获得访问所有文件权限", Toast.LENGTH_SHORT).show()
} else {
    val builder = AlertDialog.Builder(this)
        .setMessage("本程序需要您同意允许访问所有文件权限")
        .setPositiveButton("确定") { _, _ ->
            val intent = Intent(Settings.ACTION_MANAGE_ALL_FILES_ACCESS_PERMISSION)
            startActivity(intent)
        }
    builder.show()
}

亲测好用!

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐