Android 小米手机 Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES)读取文件失败。

添加图片文件到存储卡的源码如下:

   void createExternalStoragePublicPicture() {
        // Create a path where we will place our picture in the user's
        // public pictures directory.  Note that you should be careful about
        // what you place here, since the user often manages these files.  For
        // pictures and other media owned by the application, consider
        // Context.getExternalMediaDir().
        File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
        File file = new File(path, "DemoPicture.jpg");

        try {
            // Make sure the Pictures directory exists.
            path.mkdirs();

            // Very simple code to copy a picture from the application's
            // resource into the external file.  Note that this code does
            // no error checking, and assumes the picture is small (does not
            // try to copy it in chunks).  Note that if external storage is
            // not currently mounted this will silently fail.
            OutputStream os = new FileOutputStream(file);
            InputStream is = getResources().openRawResource(R.drawable.balloons);
            byte[] data = new byte[is.available()];
            is.read(data);
            os.write(data);
            is.close();
            os.close();

            // Tell the media scanner about the new file so that it is
            // immediately available to the user.
            MediaScannerConnection.scanFile(this,
                    new String[]{file.toString()}, null,
                    new MediaScannerConnection.OnScanCompletedListener() {
                        public void onScanCompleted(String path, Uri uri) {
                            Log.i("ExternalStorage", "Scanned " + path + ":");
                            Log.i("ExternalStorage", "-> uri=" + uri);
                        }
                    });
        } catch (IOException e) {
            // Unable to create file, likely because external storage is
            // not currently mounted.
            Log.w("ExternalStorage", "Error writing " + file, e);
        }
    }

 

从存取卡中读取图形文件的代码如下:

 @Override
 public boolean onOptionsItemSelected(MenuItem item) {
        File path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
        File file = new File(path, "DemoPicture.jpg");

 

在其他手机都写和读正常。独独破小米3。

然后1.权限检查。2.bug跟踪还是提示FileNoFound和EISDIR。

3.上网搜,说更新驱动。更新了也没屁用。

4.关闭USB调试。

5.root工具查看权限,还是不行。但奇怪的是我的写卡权限居然在小米中查不到。看了ES浏览器等也没有提示。估计小米屏蔽了。晕死。

5.关闭USB调试后作为MTP设备连接电脑。查看到居然有两个/storage,其中一个下面有/storage/emulated/legacy/MIUI。一个下面有/storage/emulated/0。果断删除前一个。运行程序。正常。

 

2015.4.27

删除了好多下面的音乐文件。哭。

后来采用新的方法。直接用绝对路径来保存文件。比如:file:///sdcard/1.mp3。  

<think>嗯,用户问的是关于AndroidEnvironment.getExternalStorageDirectory().getAbsolutePath() + File.separator这个文件路径的问题。首先,我需要回忆一下这个API的作用和使用情况。 首先,Environment.getExternalStorageDirectory()这个方法,我记得是用来获取外部存储的根目录的,也就是通常说的SD卡路径。不过在Android的版本演进中,这个概念可能有变化。比如在Android 10及以上,可能对访问外部存储有限制,需要用作用域存储或者其他方法。 然后,用户提到的.getAbsolutePath(),这应该是获取绝对路径的字符串。加上File.separator,可能是为了确保路径末尾有斜杠,这样在拼接子目录或文件名的时候不会出错。比如,如果原路径是/storage/emulated/0,加上File.separator后就变成/storage/emulated/0/,这样再拼接"Pictures"就会正确形成/storage/emulated/0/Pictures。 不过,需要注意的是,从Android API 29开始,这个方法被标记为过时了。现在推荐使用Context.getExternalFilesDir()或者其他方法,比如MediaStore来访问公共目录。用户可能是在维护旧代码,或者在使用较旧的教程,所以需要提醒他们关于API过时的问题,以及潜在的安全和兼容性风险。 另外,用户可能想知道这个路径具体指向哪里。一般来说,外部存储目录是用户可见的存储位置,比如手机的内部存储,而不是真正的物理SD卡。不同厂商的设备路径可能略有不同,但通常类似/storage/emulated/0或者/mnt/sdcard这样的路径。 还要考虑权限问题。在旧版本的Android中,访问外部存储需要READ_EXTERNAL_STORAGE或WRITE_EXTERNAL_STORAGE权限。如果用户的应用没有正确声明这些权限,可能会导致访问失败或者崩溃。需要提醒用户检查权限设置。 此外,用户可能在使用这个路径来保存或读取文件,比如图片、下载内容等。需要建议他们根据不同的文件类型使用对应的公共目录,比如Environment.DIRECTORY_PICTURES或者Environment.DIRECTORY_DOWNLOADS,这样系统会更合适地处理这些文件,并且在不同的Android版本上兼容性更好。 最后,可能还需要提到替代方案,比如使用Context的方法来获取应用私有目录,或者使用MediaStore来访问媒体文件,特别是在Android 10及以上版本中,直接访问外部存储目录可能不再被允许,必须通过ContentResolver进行操作。 总结一下,回答的结构应该包括:方法的作用、路径示例、过时警告、替代方案、权限问题,以及不同Android版本的差异。这样用户能全面了解这个路径的使用情况和注意事项,避免常见的坑。</think>关于 `Environment.getExternalStorageDirectory().getAbsolutePath() + File.separator` 的文件路径,以下是详细说明: --- ### 1. **作用** - 这是 Android 中获取**外部存储根目录路径**的旧方法。 - 返回的路径通常是设备的主共享存储空间(如内部存储的 `/storage/emulated/0` 或 `/sdcard`)。 - 加 `File.separator` 是为了确保路径以斜杠结尾(如 `/sdcard/`),方便后续拼接子目录。 --- ### 2. **路径示例** - **典型结果**: `"/storage/emulated/0/"` 或 `"/sdcard/"` - **拼接示例**: 若需要访问 `Pictures` 目录: ```java String path = Environment.getExternalStorageDirectory().getAbsolutePath() + File.separator + "Pictures"; // 结果:/storage/emulated/0/Pictures ``` --- ### 3. **过时警告** - **API 版本问题**: 从 **Android 10 (API 29)** 开始,此方法被标记为 `@Deprecated`。 - **限制原因**: - 直接访问共享存储可能破坏用户隐私(如访问其他应用的文件)。 - 未来版本可能完全禁止此行为。 --- ### 4. **替代方案** #### 方案 1:使用 `Context` 的专用目录(推荐) ```java // 获取应用私有目录(无需权限) File privateDir = context.getExternalFilesDir(Environment.DIRECTORY_PICTURES); // 路径示例:/storage/emulated/0/Android/data/<package>/files/Pictures ``` #### 方案 2:通过 `MediaStore` 访问公共文件Android 10+) ```java ContentResolver resolver = context.getContentResolver(); Uri uri = MediaStore.Images.Media.getContentUri(MediaStore.VOLUME_EXTERNAL); Cursor cursor = resolver.query(uri, null, null, null); // 使用 ContentResolver 读写文件 ``` #### 方案 3:请求所有文件访问权限(谨慎使用) - 在 `AndroidManifest.xml` 添加: ```xml <uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE"/> ``` - 通过 `Intent` 引导用户授权: ```java Intent intent = new Intent(Settings.ACTION_MANAGE_ALL_FILES_ACCESS_PERMISSION); startActivity(intent); ``` --- ### 5. **兼容性建议** - **API 区分处理**: ```java if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) { // 使用 MediaStore 或 Context 方法 } else { // 使用旧版 Environment.getExternalStorageDirectory() } ``` - **权限检查**: ```java if (ContextCompat.checkSelfPermission(context, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { // 请求权限 } ``` --- ### 总结 - **旧代码维护**:理解此路径的作用,但避免在新项目中使用。 - **新开发**:优先使用 `Context.getExternalFilesDir()` 或 `MediaStore`。 - **权限**:始终检查 `WRITE_EXTERNAL_STORAGE` 权限(API <29 时)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值