返回

Android 开发:轻松读写用户文件

Android

Android 10及以上版本:通过临时授权读写用户文件

前言

在Android 10及以上版本中,谷歌为保护用户隐私对文件读写权限进行了严格的控制。应用程序必须在操作用户文件之前,向用户请求必要的权限。然而,用户可能拒绝授权,导致应用程序无法正常运行。

本文将深入探讨如何通过临时授权在Android 10及以上版本中读写用户文件。它将涵盖以下主题:

  • 使用存储访问框架(SAF)
  • 请求读取/写入权限
  • 代码示例

使用存储访问框架 (SAF)

存储访问框架 (SAF) 是一种Android API,允许应用程序在用户授予临时授权后访问用户文件。它通过Intent提供了一个标准化的方式,让应用程序可以请求特定文件的访问权限。

请求读取/写入权限

要通过SAF请求读取或写入权限,请执行以下步骤:

  1. 使用 createDocumentFileopenDocumentTree 意图创建一个 DocumentFile 对象。
  2. 调用 requestPermissions 方法,传递要请求的权限和请求代码。
  3. onRequestPermissionsResult 回调方法中,检查权限是否被授予。

代码示例

以下代码示例演示了如何通过SAF请求读取用户文件的权限:

import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.provider.DocumentsContract;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

public class MainActivity extends AppCompatActivity {

    private static final int REQUEST_READ_EXTERNAL_STORAGE_PERMISSION = 1;

    private TextView tvFileContent;
    private Button btnReadFile;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        tvFileContent = findViewById(R.id.tvFileContent);
        btnReadFile = findViewById(R.id.btnReadFile);

        btnReadFile.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                // 检查是否具有读取外部存储的权限
                if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.READ_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) {
                    // 如果已经具有权限,则直接读取文件
                    readFile();
                } else {
                    // 如果没有权限,则请求权限
                    ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, REQUEST_READ_EXTERNAL_STORAGE_PERMISSION);
                }
            }
        });
    }

    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);

        switch (requestCode) {
            case REQUEST_READ_EXTERNAL_STORAGE_PERMISSION:
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                    // 用户授权了读取外部存储的权限
                    readFile();
                } else {
                    // 用户拒绝了读取外部存储的权限
                    Log.e("MainActivity", "用户拒绝了读取外部存储的权限");
                }
                break;
        }
    }

    private void readFile() {
        // 创建 SAF Intent
        Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
        intent.addCategory(Intent.CATEGORY_OPENABLE);
        intent.setType("*/*");

        // 启动 SAF
        startActivityForResult(intent, REQUEST_READ_EXTERNAL_STORAGE_PERMISSION);
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);

        if (requestCode == REQUEST_READ_EXTERNAL_STORAGE_PERMISSION && resultCode == RESULT_OK) {
            // 获取文件内容
            Uri uri = data.getData();
            try {
                String fileContent = readTextFromUri(uri);
                tvFileContent.setText(fileContent);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    private String readTextFromUri(Uri uri) throws IOException {
        try (InputStream inputStream = getContentResolver().openInputStream(uri)) {
            BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
            StringBuilder builder = new StringBuilder();
            String line;
            while ((line = reader.readLine()) != null) {
                builder.append(line).append("\n");
            }
            return builder.toString();
        }
    }
}

常见问题解答

1. 如何在没有用户交互的情况下请求权限?

您可以使用 requestPermissions 方法并设置 shouldShowRationale 参数为 false。这将允许您在没有用户交互的情况下请求权限,但用户仍可以在设置中撤销权限。

2. 如何处理权限被拒绝的情况?

如果您请求的权限被拒绝,您可以显示一个解释性消息,告知用户为什么需要该权限。您还可以提供一个按钮,让用户重新授予权限。

3. SAF 与 MediaStore 之间有什么区别?

MediaStore 是一种提供访问媒体文件(如图片、视频和音频)的API。它提供了与 SAF 类似的功能,但仅适用于媒体文件。

4. 临时授权是否始终可用?

临时授权并不总是可用。如果用户之前拒绝了该权限,或者如果该权限是危险权限(如位置或联系人),则系统可能不会提供临时授权。

5. 我可以用 SAF 访问所有文件吗?

SAF 不允许您访问所有文件。某些类型的文件,如系统文件,可能会被排除在外。

结论

通过临时授权读写用户文件是Android 10及以上版本中保护用户隐私和确保应用程序正常运行的关键。本文提供了有关如何使用存储访问框架 (SAF) 请求读取/写入权限的详细指南。通过遵循本文中的步骤,您可以为您的应用程序构建安全、用户友好的文件访问功能。