0

0

解决 Android 外部存储权限导致的文件访问异常

心靈之曲

心靈之曲

发布时间:2025-07-20 14:26:16

|

1238人浏览过

|

来源于php中文网

原创

解决 android 外部存储权限导致的文件访问异常

本文旨在深入解析 Android 应用中常见的 FileNotFoundException: EACCES (Permission denied) 错误,特别是在访问外部存储文件时遇到的权限问题。我们将探讨 Android 存储权限模型的演进,包括分区存储(Scoped Storage)的影响,并提供详细的权限配置、运行时请求以及针对 Android 11+ 版本的解决方案,确保应用能够正确、安全地访问外部文件。

理解 Android 存储权限错误:EACCES

当 Android 应用尝试访问外部存储(如 /storage/emulated/0/Download/)中的文件,但缺乏必要的权限时,就会抛出 java.io.FileNotFoundException: /path/to/file.xlsx: open failed: EACCES (Permission denied) 异常。这里的 EACCES 明确指示了“访问被拒绝”,意味着操作系统出于安全考虑,阻止了应用对指定路径的读写操作。

此问题的根源在于 Android 系统对外部存储访问权限的严格控制和不断演进的隐私策略。从 Android 6.0 (Marshmallow) 开始引入的运行时权限,到 Android 10 (Q) 和 Android 11 (R) 引入的分区存储(Scoped Storage),都对应用访问外部存储的方式提出了更高的要求。特别是分区存储,它旨在限制应用只能访问自身创建的文件或特定媒体类型的文件,除非获得特殊权限。

核心权限声明与配置

为了访问外部存储,应用需要在 AndroidManifest.xml 中声明相应的权限。根据 Android 版本的不同,所需的权限和配置也有所区别。

  1. 基础外部存储权限 (Android 6.0 - Android 10): 对于 Android 6.0 (API 23) 到 Android 10 (API 29) 的设备,READ_EXTERNAL_STORAGE 和 WRITE_EXTERNAL_STORAGE 是访问外部存储的常用权限。

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
  2. 针对 Android 10 的兼容性处理 (requestLegacyExternalStorage): 在 Android 10 (API 29) 中,Google 引入了分区存储。为了让现有应用能够平滑过渡,可以设置 android:requestLegacyExternalStorage="true" 来暂时禁用分区存储,恢复旧版存储模型。

    <application
        android:requestLegacyExternalStorage="true"
        ...>
        <!-- ... -->
    </application>

    注意: requestLegacyExternalStorage 在 Android 11 及更高版本中不再有效。

  3. 针对 Android 11 及更高版本的全文件访问权限 (MANAGE_EXTERNAL_STORAGE): 从 Android 11 (API 30) 开始,如果应用需要访问设备上的所有文件(例如文件管理器、备份工具等),则必须声明 MANAGE_EXTERNAL_STORAGE 权限。此权限被称为“所有文件访问权限”,并需要用户手动授权。

    <uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE"
                     tools:ignore="ScopedStorage"/>

    tools:ignore="ScopedStorage" 属性用于抑制 Lint 警告,表明你清楚正在请求此特殊权限。

    完整的 Manifest 示例: 结合上述权限,一个典型的 AndroidManifest.xml 片段可能如下所示:

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
              xmlns:tools="http://schemas.android.com/tools"
              package="com.prod.gfer94">
    
        <!-- 基础外部存储读写权限,适用于Android 10及以下,在Android 11+可能受限 -->
        <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
        <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    
        <!-- 针对Android 11+,如果需要访问所有文件,必须声明此权限 -->
        <uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE"
                         tools:ignore="ScopedStorage"/>
    
        <application
            android:icon="@drawable/icogaef"
            android:label="@string/app_name"
            android:roundIcon="@drawable/icogaef"
            android:theme="@style/AppTheme"
            android:requestLegacyExternalStorage="true"> <!-- 兼容Android 10 -->
            <!-- ... -->
        </application>
    </manifest>

    请注意,minSdkVersion 和 targetSdkVersion 对权限行为有重要影响。建议将 targetSdkVersion 更新到最新稳定版本,并根据其行为调整权限处理逻辑。

运行时权限请求

仅仅在 AndroidManifest.xml 中声明权限是不够的。对于危险权限(如 READ_EXTERNAL_STORAGE 和 WRITE_EXTERNAL_STORAGE),从 Android 6.0 (API 23) 开始,应用必须在运行时向用户请求授权。

以下是请求外部存储读写权限的基本流程:

Chromox
Chromox

Chromox是一款领先的AI在线生成平台,专为喜欢AI生成技术的爱好者制作的多种图像、视频生成方式的内容型工具平台。

下载
  1. 检查权限是否已授予: 使用 ContextCompat.checkSelfPermission() 方法检查当前是否已获得权限。

  2. 请求权限: 如果权限尚未授予,使用 ActivityCompat.requestPermissions() 方法向用户发起权限请求。

  3. 处理权限请求结果: 在 onRequestPermissionsResult() 回调方法中处理用户的选择。

示例代码:

import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Environment;
import android.provider.Settings;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;

// ... 在你的Activity中

private static final int REQUEST_CODE_STORAGE_PERMISSION = 101;
private static final int REQUEST_CODE_MANAGE_ALL_FILES_ACCESS = 102;

/**
 * 检查并请求存储权限。
 * 如果是 Android 11+ 且需要所有文件访问,会引导用户到设置页面。
 */
private void checkStoragePermissions() {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) { // Android 11 (API 30) 及更高版本
        if (Environment.isExternalStorageManager()) {
            // 已获得所有文件访问权限
            performFileOperation();
        } else {
            // 请求所有文件访问权限
            requestManageAllFilesAccess();
        }
    } else { // Android 10 (API 29) 及以下版本
        if (ContextCompat.checkSelfPermission(this, Manifest.permission.READ_EXTERNAL_STORAGE)
                == PackageManager.PERMISSION_GRANTED &&
            ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE)
                == PackageManager.PERMISSION_GRANTED) {
            // 已获得读写权限
            performFileOperation();
        } else {
            // 请求读写权限
            ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE},
                    REQUEST_CODE_STORAGE_PERMISSION);
        }
    }
}

/**
 * 请求 MANAGE_EXTERNAL_STORAGE 权限 (Android 11+)
 */
private void requestManageAllFilesAccess() {
    try {
        android.content.Intent intent = new android.content.Intent(Settings.ACTION_MANAGE_APP_ALL_FILES_ACCESS_PERMISSION);
        intent.setData(android.net.Uri.parse("package:" + getPackageName()));
        startActivityForResult(intent, REQUEST_CODE_MANAGE_ALL_FILES_ACCESS);
    } catch (Exception e) {
        // 如果无法启动特定应用设置页面,则尝试启动所有文件访问设置页面
        android.content.Intent intent = new android.content.Intent(Settings.ACTION_MANAGE_ALL_FILES_ACCESS_PERMISSION);
        startActivityForResult(intent, REQUEST_CODE_MANAGE_ALL_FILES_ACCESS);
    }
}

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    if (requestCode == REQUEST_CODE_STORAGE_PERMISSION) {
        if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
            // 权限已授予
            performFileOperation();
        } else {
            // 权限被拒绝,可以提示用户或禁用相关功能
            android.widget.Toast.makeText(this, "存储权限被拒绝,无法访问文件。", android.widget.Toast.LENGTH_SHORT).show();
        }
    }
}

@Override
protected void onActivityResult(int requestCode, int resultCode, android.content.Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (requestCode == REQUEST_CODE_MANAGE_ALL_FILES_ACCESS) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
            if (Environment.isExternalStorageManager()) {
                // 再次检查权限,确保用户已授权
                performFileOperation();
            } else {
                android.widget.Toast.makeText(this, "所有文件访问权限被拒绝,无法访问文件。", android.widget.Toast.LENGTH_SHORT).show();
            }
        }
    }
}

/**
 * 实际执行文件操作的方法
 */
private void performFileOperation() {
    String filePath = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS).getAbsolutePath() + "/Joueurs.xlsx";
    java.io.File fileName = new java.io.File(filePath);

    if (!fileName.exists()) {
        android.widget.Toast.makeText(this, "文件不存在: " + filePath, android.widget.Toast.LENGTH_LONG).show();
        return;
    }

    try {
        if (!fileName.canRead()) {
            // 尝试设置文件可读(在某些情况下可能有效,但权限不足时仍会失败)
            fileName.setReadable(true);
        }

        java.io.FileInputStream fileJoueur = new java.io.FileInputStream(fileName);
        org.apache.poi.xssf.usermodel.XSSFWorkbook myWorkBook = new org.apache.poi.xssf.usermodel.XSSFWorkbook(fileJoueur);
        org.apache.poi.xssf.usermodel.XSSFSheet mySheet = myWorkBook.getSheetAt(0);
        // ... 继续处理 Excel 文件
        android.widget.Toast.makeText(this, "成功读取Excel文件!", android.widget.Toast.LENGTH_SHORT).show();
        fileJoueur.close();
        myWorkBook.close();
    } catch (java.io.FileNotFoundException e) {
        android.util.Log.e("FileAccess", "文件未找到或权限不足: " + e.getMessage(), e);
        android.widget.Toast.makeText(this, "文件未找到或权限不足: " + e.getMessage(), android.widget.Toast.LENGTH_LONG).show();
    } catch (java.io.IOException e) {
        android.util.Log.e("FileAccess", "文件读取错误: " + e.getMessage(), e);
        android.widget.Toast.makeText(this, "文件读取错误: " + e.getMessage(), android.widget.Toast.LENGTH_LONG).show();
    }
}

// 在你的按钮点击事件中调用
// BoutonLicencies.setOnClickListener(v -> checkStoragePermissions());

处理 Android 11 及更高版本的文件访问

对于 Android 11 (API 30) 及更高版本,分区存储是默认行为。这意味着即使声明了 READ_EXTERNAL_STORAGE 和 WRITE_EXTERNAL_STORAGE,应用也只能访问:

  • 应用私有目录 (Context.getExternalFilesDir(), Context.getExternalCacheDir())。
  • 通过 MediaStore API 访问的媒体文件(图片、视频、音频),且仅限于应用自身创建的或用户明确授予访问权限的。
  • 通过 Storage Access Framework (SAF) 选择的文件。

如果应用确实需要访问外部存储中的任意文件(例如用户下载的 Excel 文件),则 MANAGE_EXTERNAL_STORAGE 权限是目前最直接的解决方案。然而,获取此权限需要用户手动导航到设置页面进行授权,并且 Google Play 对使用此权限的应用有严格的审查政策。

推荐方案:使用 Storage Access Framework (SAF) 或 MediaStore API

在大多数情况下,为了遵循 Android 的最佳实践和用户隐私原则,推荐使用 SAF 或 MediaStore API 来访问文件,而不是请求广范围的存储权限。

  • Storage Access Framework (SAF): SAF 允许用户通过系统文件选择器授予应用对特定文件或目录的临时访问权限,而无需应用拥有全局存储权限。这是访问用户下载目录中特定文件的推荐方式。

    // 启动文件选择器
    private static final int PICK_EXCEL_FILE_REQUEST = 1;
    
    private void openFilePicker() {
        android.content.Intent intent = new android.content.Intent(android.content.Intent.ACTION_OPEN_DOCUMENT);
        intent.addCategory(android.content.Intent.CATEGORY_OPENABLE);
        intent.setType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"); // for .xlsx
        // intent.setType("application/vnd.ms-excel"); // for .xls
        startActivityForResult(intent, PICK_EXCEL_FILE_REQUEST);
    }
    
    @Override
    protected void onActivityResult(int requestCode, int resultCode, android.content.Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == PICK_EXCEL_FILE_REQUEST && resultCode == RESULT_OK) {
            if (data != null) {
                android.net.Uri uri = data.getData();
                try {
                    // 使用 ContentResolver 获取文件输入流
                    java.io.InputStream inputStream = getContentResolver().openInputStream(uri);
                    org.apache.poi.xssf.usermodel.XSSFWorkbook myWorkBook = new org.apache.poi.xssf.usermodel.XSSFWorkbook(inputStream);
                    org.apache.poi.xssf.usermodel.XSSFSheet mySheet = myWorkBook.getSheetAt(0);
                    // ... 处理 Excel 文件
                    android.widget.Toast.makeText(this, "成功通过SAF读取Excel文件!", android.widget.Toast.LENGTH_SHORT).show();
                    inputStream.close();
                    myWorkBook.close();
                } catch (java.io.FileNotFoundException e) {
                    android.util.Log.e("SAF_FileAccess", "文件未找到: " + e.getMessage(), e);
                    android.widget.Toast.makeText(this, "文件未找到或无法打开: " + e.getMessage(), android.widget.Toast.LENGTH_LONG).show();
                } catch (java.io.IOException e) {
                    android.util.Log.e("SAF_FileAccess", "文件读取错误: " + e.getMessage(), e);
                    android.widget.Toast.makeText(this, "文件读取错误: " + e.getMessage(), android.widget.Toast.LENGTH_LONG).show();
                }
            }
        }
    }
    // 在你的按钮点击事件中调用
    // BoutonLicencies.setOnClickListener(v -> openFilePicker());
  • MediaStore API: 适用于访问图片、视频、音频等媒体文件。它提供了一种结构化的方式来查询和操作这些文件,而无需直接的文件路径访问。对于 Excel 文件,SAF 是更合适的选择。

代码实践与注意事项

  1. 文件路径的正确获取: 避免硬编码 /storage/emulated/0/Download/ 这样的路径。应使用 Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS) 来获取下载目录的标准路径,这更具兼容性。

  2. 权限检查与错误处理: 在进行任何文件操作之前,务必检查是否已获得所需权限。捕获 FileNotFoundException 和 IOException 以优雅地处理文件不存在、权限不足或读写错误的情况。

  3. 针对特定设备(如 Galaxy S21)的兼容性考虑: 某些设备制造商可能会对 Android 系统进行定制,导致权限设置界面或行为略有不同。例如,Galaxy S21 运行 Android 11 或更高版本时,如果应用未正确处理分区存储和 MANAGE_EXTERNAL_STORAGE 权限,用户可能无法在设置中找到“所有文件”的权限选项,或者即使找到了也无法正常工作。在这种情况下,确保 targetSdkVersion 匹配设备版本,并正确实现 MANAGE_EXTERNAL_STORAGE 的请求流程至关重要。

  4. 更新 targetSdkVersion 的重要性: 将 targetSdkVersion 更新到最新版本,可以确保应用的行为与最新 Android 系统的安全和隐私特性保持一致。虽然这可能需要修改代码以适应新的行为,但从长远来看,这有助于提高应用的兼容性和安全性。

总结

解决 Android FileNotFoundException: EACCES 错误的关键在于深入理解 Android 存储权限模型的演变,并根据应用的 targetSdkVersion 和所需的访问级别,正确声明和请求权限。对于 Android 11 及更高版本,优先考虑使用 Storage Access Framework (SAF) 进行文件访问,以符合分区存储的最佳实践。如果确实需要广泛的文件访问权限,则应正确实现 MANAGE_EXTERNAL_STORAGE 的请求流程,并准备好应对 Google Play 的审核要求。通过细致的权限管理和错误处理,可以确保应用在不同 Android 版本和设备上稳定、安全地访问外部文件。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1948

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2119

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1168

2024.11.28

android开发三大框架
android开发三大框架

android开发三大框架是XUtil框架、volley框架、ImageLoader框架。本专题为大家提供android开发三大框架相关的各种文章、以及下载和课程。

338

2023.08.14

android是什么系统
android是什么系统

Android是一种功能强大、灵活可定制、应用丰富、多任务处理能力强、兼容性好、网络连接能力强的操作系统。本专题为大家提供android相关的文章、下载、课程内容,供大家免费下载体验。

1819

2023.08.22

android权限限制怎么解开
android权限限制怎么解开

android权限限制可以使用Root权限、第三方权限管理应用程序、ADB命令和Xposed框架解开。详细介绍:1、Root权限,通过获取Root权限,用户可以解锁所有权限,并对系统进行自定义和修改;2、第三方权限管理应用程序,用户可以轻松地控制和管理应用程序的权限;3、ADB命令,用户可以在设备上执行各种操作,包括解锁权限;4、Xposed框架,用户可以在不修改系统文件的情况下修改应用程序的行为和权限。

2137

2023.09.19

android重启应用的方法有哪些
android重启应用的方法有哪些

android重启应用有通过Intent、PendingIntent、系统服务、Runtime等方法。本专题为大家提供Android相关的文章、下载、课程内容,供大家免费下载体验。

284

2023.10.18

Android语音播放功能实现方法
Android语音播放功能实现方法

实现方法有使用MediaPlayer实现、使用SoundPool实现两种。可以根据具体的需求选择适合的方法进行实现。想了解更多语音播放的相关内容,可以阅读本专题下面的文章。

380

2024.03.01

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
RunnerGo从入门到精通
RunnerGo从入门到精通

共22课时 | 1.8万人学习

尚学堂Mahout视频教程
尚学堂Mahout视频教程

共18课时 | 3.3万人学习

Linux优化视频教程
Linux优化视频教程

共14课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号