uni-app 真机调试安卓11设备读写失败问题

uni-app 真机调试安卓11设备读写失败问题

### 项目信息

| 信息           |                       |
|----------------|-------------------------|
| 产品分类       | uniapp/App              |
| PC开发环境     | Windows                 |
| PC开发环境版本 | Windows 10 专业版       |
| HBuilderX类型  | Alpha                   |
| HBuilderX版本  | 3.4.9                   |
| 手机系统       | Android                 |
| 手机系统版本   | Android 11              |
| 手机厂商       | Pixel 5                 |
| 手机机型       | Pixel 5                 |
| 页面类型       | vue                     |
| vue版本        | vue2                    |
| 打包方式       | 云端                    |
| 项目创建方式   | HBuilderX               |

### 操作步骤
- 检测到正在运行真机调试的是安卓11设备,可能会出现读写失败问题,如遇到此类问题

### 预期结果
- 检测到正在运行真机调试的是安卓11设备,可能会出现读写失败问题,如遇到此类问题

### 实际结果
- 检测到正在运行真机调试的是安卓11设备,可能会出现读写失败问题,如遇到此类问题

### bug描述
- 检测到正在运行真机调试的是安卓11设备,可能会出现读写失败问题,如遇到此类问题  
  push: \unpackage\dist\dev\app-plus/./app-config-service.js -> /sdcard/DCloud/temp/android.com/apps/UNI5013B7D/Command-0/www/app-config-service.jsard/DCloud/temp/androidwxc.hosian.com/apps/UNI5013B7D/Command-0/www/androidPrivacy.json
2 回复

我也遇到了 求怎么解决


在 uni-app 开发过程中,真机调试时遇到安卓11设备读写失败的问题,通常是由于安卓11(API 30)引入的存储访问权限变更(Scoped Storage)导致的。Scoped Storage 限制了应用对文件系统的直接访问,尤其是对公共存储区域的访问。

解决方案

1. 配置 AndroidManifest.xml

AndroidManifest.xml 文件中,添加以下权限声明:

<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" />

2. 请求存储权限

在应用启动时,动态请求存储权限:

// 在 App.vue 或 main.js 中
uni.getSystemInfo({
  success: function (res) {
    if (res.platform === 'android' && res.system.indexOf('11') !== -1) {
      uni.authorize({
        scope: 'scope.writePhotosAlbum',
        success: function () {
          console.log('授权成功');
        },
        fail: function () {
          console.log('授权失败');
        }
      });
    }
  }
});

3. 使用 MANAGE_EXTERNAL_STORAGE 权限

对于需要访问所有外部存储文件的应用,可以请求 MANAGE_EXTERNAL_STORAGE 权限:

uni.request({
  url: 'android.settings.MANAGE_ALL_FILES_ACCESS_PERMISSION',
  success: function (res) {
    console.log('请求成功');
  },
  fail: function (err) {
    console.log('请求失败', err);
  }
});

4. 使用 FileProvider

如果需要访问应用私有目录以外的文件,可以使用 FileProvider

<provider
    android:name="androidx.core.content.FileProvider"
    android:authorities="${applicationId}.fileprovider"
    android:exported="false"
    android:grantUriPermissions="true">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"
        android:resource="@xml/file_paths" />
</provider>

res/xml/file_paths.xml 中定义文件路径:

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="." />
</paths>

5. 使用 MediaStore

对于媒体文件的访问,可以使用 MediaStore API:

const uri = android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
const cursor = uni.getContentResolver().query(uri, null, null, null, null);
if (cursor != null) {
    while (cursor.moveToNext()) {
        const displayName = cursor.getString(cursor.getColumnIndex(android.provider.MediaStore.Images.Media.DISPLAY_NAME));
        console.log(displayName);
    }
    cursor.close();
}
回到顶部
AI 助手
你好,我是IT营的 AI 助手
您可以尝试点击下方的快捷入口开启体验!