在进行需求开发的时分,咱们总是避不开和用户的数据打交道,那提到获取用户的数据一定会想到的东西便是请求权限

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

在我刚学习安卓的时分,我是以为APP一定要声明了读写用户空间权限而且在用户授权之后才能获取到用户的文件,即使是做个简简单单的更换头像的功能,或许是晋级APP时下载新的APK。关于后者,咱们其实能够将晋级的APK包放到咱们运用的私有目录下(无需权限),关于前者,有什么比较轻量,适合快速开发需求的办法来满足呢。

这儿插三段小阐明,假如只想知道办法的能够直接越过
  • 首要咱们要理解,为什么谷歌要用读写权限来限制APP对用户文件的操作权。答案其实很明显,由于需求避免APP歹意侵略用户隐私,或许是在用户的目录里寄存很多的垃圾文件,在用户目录里寄存的文件是不会随着APP的卸载而被删去的,所以假如所有APP都在用户的目录里寄存文件(像是相册文件夹/下载文件夹),那用户的体会别提有多糟糕了。

  • 其次便是声明权限其实是有挺多弊端的,假如不对错有必要的权限,其实谷歌是期望咱们能不要就不要的。做过谷歌运用商场开发的就知道,你声明的每个权限都会在谷歌运用的详情页标示,这不仅仅是让用户一进来就觉得:”这个APP又要窥视我隐私”,而且是让你在填运用的数据安全表单时愈加地费事,由于你声明了读写权限,那你就要阐明你的APP会获取用户的什么数据,怎么保存,用户是否能够删去以及是否知情等等。还有便是你声明的权限越多,你的运用审核时间就会越长,这个我信任没有人觉得无所谓吧

  • 第三便是,Android11及以上的版别其实现已大削了WRITE_EXTERNAL_STORAGE这个权限,谷歌不再允许APP悄悄地在用户的外置存储目录里偷偷拉大便了,你在用户目录里创立什么目录存取什么数据都要在用户知情而且同意的情况下才能进行,而本文要介绍的办法是能兼容到Android13的,所以赶紧学起来吧^-^

接下来是正文,首要我模仿获取用户的图片的逻辑
  1. 咱们需求拿到代表用户暂时授权给APP的Uri

    经过

    val intent = Intent(Intent.ACTION_GET_CONTENT)
        .addCategory(Intent.CATEGORY_OPENABLE)
        //这儿传的参数是你要获取的文件类型的mimeType
        .setType(mimeType)
    startActivityForResult(intent,1024)
    override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
        super.onActivityResult(requestCode, resultCode, data)
        if (requestCode == 1024 && resultCode == RESULT_OK) {
            val uri = data?.data
            //这儿获取到的uri便是用户暂时授权的文件/文件夹的的标识
        }
    }
    

    或许

    val launch = registerForActivityResult(ActivityResultContracts.GetContent()){uri->
    //这儿获取到的uri便是用户暂时授权的文件/文件夹的的标识
    }
    //这儿传的参数是你要获取的文件类型的mimeType
    launch.launch("*/*")
    

    发动体系的内容选择器让用户选择要分享给咱们APP的文件,以获得文件的Uri

  2. 经过contentResolver翻开文件的文件描述符FileDescriptor

    val pfd : ParcelFileDescriptor? = context.contentResolver.openFileDescriptor(uri, "r")
    

    第一个参数是咱们刚刚得到的文件的uri,第二个文件是表明咱们对文件的操作形式,我现在演示的是读取用户图片所以用只读形式(“r”)就能够了,关于mode的具体注释,这儿我直接张贴原文了

    mode – The string representation of the file mode. Can be "r", "w", "wt", "wa", "rw" or "rwt". SeeParcelFileDescriptor.parseMode for more details.

  3. 经过FileDescriptor能够翻开一个文件IO流(FIS或许FOS),就能够读写文件啦

    FileInputStream(pfd.fileDescriptor).use {
    //这儿能够先将用户的图片复制到私有目录中,再让用户做进一步的编辑操作
    }
    FileOutputStream(pfd.fileDescriptor).use {
    }
    

    可是留意,翻开的fileDescriptor是Closeable对象,所以用完之后需求手动close(),这儿我用的是ktolin的扩展函数,会在use代码块里的代码运行完之后自动封闭流

    另一种读取文件的办法,仍是运用contentResolver直接翻开io流

    context.contentResolver.openInputStream(uri)?.use {
    }
    context.contentResolver.openOutputStream(uri)?.use {
    }
    
接下来再模仿一下将文件写入用户目录的操作

其实思路是如出一辙的,只是你发动文件体系的目的(intent)不一样,以及对文件的操作不一样

  1. 咱们需求拿到代表用户暂时授权给APP的Uri

    //这儿传入你要创立的文件类型的mimeType,假如是"*/*"那就代表文件夹
    val launcher = registerForActivityResult(ActivityResultContracts.CreateDocument("*/*")){uri->
        //这儿获取到的uri是现已创立好的文件的uri
    }
    //这儿传入要创立的文件名
    launcher.launch("cache.png")
    

    发动之后是这个界面

    Android无需读写权限(通过用户临时授权)读写用户的文件
  2. 经过contentResolver翻开文件的文件描述符FileDescriptor

    val pfd : ParcelFileDescriptor? = context.contentResolver.openFileDescriptor(uri, "rw")
    

    第一个参数是咱们刚刚得到的文件的uri,第二个文件是表明咱们对文件的操作形式,我现在演示的是保存一张图片所以要用读写形式(“rw”)

  3. 经过FileDescriptor能够翻开一个文件IO流(FIS或许FOS),就能够写文件啦

    FileOutputStream(pfd.fileDescriptor).use {
    //这儿将处理好的图片利用fos写到用户方才用uri指定的地方
    }
    

    另一种读取文件的办法,仍是运用contentResolver直接翻开io流

    context.contentResolver.openOutputStream(uri)?.use {
    }
    
最终再模仿一下获取用户文件夹控制权的操作,经过这个办法你能够拿到其他运用在外置存储里的目录(例如一些聊天软件的聊天记录其实便是寄存在这个目录的)

(先截了张图,过两天填坑)

Android无需读写权限(通过用户临时授权)读写用户的文件
最终再介绍一 经过Uri获取文件信息(文件名/文件巨细/文件Mime类型)的办法
//第二个参数相当所以sql里的select,列表里是要过滤的列名,假如传null那阐明取所有的列,这样功能会比较差
val cursor: Cursor? = context.contentResolver.query(
    this,
    arrayOf(MediaStore.MediaColumns.DISPLAY_NAME, MediaStore.MediaColumns.SIZE),
    null,
    null,
    null
)?.use { cursor ->
    if (cursor.moveToFirst()) {
        val columnIndex1 = cursor.getColumnIndex(MediaStore.MediaColumns.DISPLAY_NAME)
        if (columnIndex1 > -1) {
            name = cursor.getString(columnIndex1)
        }
        val columnIndex2 = cursor.getColumnIndex(MediaStore.MediaColumns.SIZE)
        if (columnIndex2 > -1) {
            size = cursor.getLong(columnIndex2)
        }
    }

文件的话,用正常途径也只能拿到文件名(MediaStore.MediaColumns.DISPLAY_NAME),文件巨细(MediaStore.MediaColumns.SIZE),文件Mime类型(MediaStore.MediaColumns.MIME_TYPE)这三个有用的信息 留意,获取到的cursor是Closeable对象,所以用完之后需求手动close()