Android 打开带有 ACTION_GET_CONTENT 的文件会导致不同的 Uri

新手上路,请多包涵

我正在尝试使用 Intent.ACTION_GET_CONTENT 打开文件。 安卓文件浏览器

根据文件浏览器打开的 Android 版本/设备品牌,我得到以下结果:

Downloads 中选择一个文件:

 content://com.android.providers.downloads.documents/document/446

Fotos 选择文件:

 content://media/external/images/media/309

FileCommander 选择文件:

 file:///storage/emulated/0/DCIM/Camera/20141027_132114.jpg

我可以打开所有这些文件,除非我尝试从 Downloads, , Audio , Afbeeldingen 打开文件(图片)

我可能无法处理这种 Uri: content://com.android.providers.downloads.documents/document/446

我尝试了以下内容:

  • 尝试通过 new File(uri.getPath()) 打开文件。 File.exists() 返回 false。
  • 尝试通过 getContext().getContentResolver().openInputStream(uri) 打开/访问文件。结果变成 FileNotFoundException
  • 尝试使用以下代码打开文件:
   public static String getRealPathFromURI_API19(Context context, Uri uri) {

  Log.i("uri", uri.getPath());
  String filePath = "";
  if (uri.getScheme().equals("file")) {
      return uri.getPath();
  } else if (DocumentsContract.isDocumentUri(context, uri)) {
      String wholeID = DocumentsContract.getDocumentId(uri);
      Log.i("wholeID", wholeID);
      // Split at colon, use second item in the array
      String[] splits = wholeID.split(":");
  if (splits.length == 2) {
      String id = splits[1];

          String[] column = {MediaStore.Images.Media.DATA};
      // where id is equal to
          String sel = MediaStore.Images.Media._ID + "=?";
          Cursor cursor = context.getContentResolver().query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
                  column, sel, new String[]{id}, null);
          int columnIndex = cursor.getColumnIndex(column[0]);
          if (cursor.moveToFirst()) {
              filePath = cursor.getString(columnIndex);
          }
          cursor.close();
      }
  } else {
      filePath = AttachmentUtils.getPath(context, uri);
  }
  return filePath;
  }

我究竟做错了什么?

更新:我注意到屏幕截图中列出的文件实际上并不存在于存储中。我使用的设备是该公司的平板电脑,其中包含垃圾数据。我的同事告诉我,这个设备曾经与另一个谷歌账户相关联。这些文件可能是以前帐户中不再存在/无法访问的文件。

我自己的结论是我在 Android 中遇到了一些错误。我的 错误报告

2017 年 2 月 6 日更新:

Android 禁止 file:// URI。请考虑考虑。

禁止文件:Uri 方案 到目前为止,Android 7.0 最大的兼容性问题是文件:Uri 值的方案实际上被禁止了。如果您尝试传递一个文件:Intent 中的 Uri,该 Intent 将传递给另一个应用程序——无论是通过额外的还是作为 Intent 的“数据”方面——您将因 FileUriExposedException 异常而崩溃。将 file: Uri 值放在剪贴板上的 ClipData 中时,您将面临类似的问题。这来自 StrictMode 的更新版本。 StrictMode.VmPolicy.Builder 有一个 penaltyDeathOnFileUriExposure() 方法触发检测文件:Uri 值和由此产生的 FileUriExposedException 异常。而且,这似乎是预先配置的,就像预先配置 StrictMode 以应用 penaltyDeathOnNetwork() 的方式一样(您的 NetworkOnMainThreadException 崩溃的来源)。

原文由 com2ghz 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 1k
2 个回答

使用下面的代码。这肯定会起作用。

 public static String getPath(Context context, Uri uri) {

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
        // DocumentProvider
        if (DocumentsContract.isDocumentUri(context, uri)) {
            // ExternalStorageProvider
            if (isExternalStorageDocument(uri)) {
                final String docId = DocumentsContract.getDocumentId(uri);
                final String[] split = docId.split(":");
                final String type = split[0];

                if ("primary".equalsIgnoreCase(type)) {
                    return Environment.getExternalStorageDirectory() + "/" + split[1];
                }
                // TODO handle non-primary volumes
            }
            // DownloadsProvider
            else if (isDownloadsDocument(uri)) {
                final String id = DocumentsContract.getDocumentId(uri);
                final Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));
                return getDataColumn(context, contentUri, null, null);
            }
            // MediaProvider
            else if (isMediaDocument(uri)) {
                final String docId = DocumentsContract.getDocumentId(uri);
                final String[] split = docId.split(":");
                final String type = split[0];
                Uri contentUri = null;
                if ("image".equals(type)) {
                    contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
                } else if ("video".equals(type)) {
                    contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
                } else if ("audio".equals(type)) {
                    contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
                }
                final String selection = "_id=?";
                final String[] selectionArgs = new String[]{split[1]};
                return getDataColumn(context, contentUri, selection, selectionArgs);
            }
        }
    }
    // MediaStore (and general)
    else if ("content".equalsIgnoreCase(uri.getScheme())) {
        // Return the remote address
        if (isGooglePhotosUri(uri))
            return uri.getLastPathSegment();
        return getDataColumn(context, uri, null, null);
    }
    // File
    else if ("file".equalsIgnoreCase(uri.getScheme())) {
        return uri.getPath();
    }
    return null;
}

public static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {
    Cursor cursor = null;
    final String column = "_data";
    final String[] projection = {column};
    try {
        cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null);
        if (cursor != null && cursor.moveToFirst()) {
            final int index = cursor.getColumnIndexOrThrow(column);
            return cursor.getString(index);
        }
    } finally {
        if (cursor != null)
            cursor.close();
    }
    return null;
}

public static boolean isExternalStorageDocument(Uri uri) {
    return "com.android.externalstorage.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is DownloadsProvider.
 */
public static boolean isDownloadsDocument(Uri uri) {
    return "com.android.providers.downloads.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is MediaProvider.
 */
public static boolean isMediaDocument(Uri uri) {
    return "com.android.providers.media.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is Google Photos.
 */
public static boolean isGooglePhotosUri(Uri uri) {
    return "com.google.android.apps.photos.content".equals(uri.getAuthority());
}

使用以下代码浏览任何格式的文件。

 public void browseClick() {

    Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
    intent.setType("*/*");
    intent.addCategory(Intent.CATEGORY_OPENABLE);
    //intent.putExtra("browseCoa", itemToBrowse);
    //Intent chooser = Intent.createChooser(intent, "Select a File to Upload");
    try {
        //startActivityForResult(chooser, FILE_SELECT_CODE);
        startActivityForResult(Intent.createChooser(intent, "Select a File to Upload"),FILE_SELECT_CODE);
    } catch (Exception ex) {
        System.out.println("browseClick :"+ex);//android.content.ActivityNotFoundException ex
    }
}

然后在 onActivityResult 中获取该文件路径,如下所示。

 @Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (requestCode == FILE_SELECT_CODE) {
        if (resultCode == RESULT_OK) {
            try {
              Uri uri = data.getData();

                if (filesize >= FILE_SIZE_LIMIT) {
                    Toast.makeText(this,"The selected file is too large. Selet a new file with size less than 2mb",Toast.LENGTH_LONG).show();
                } else {
                    String mimeType = getContentResolver().getType(uri);
                    if (mimeType == null) {
                        String path = getPath(this, uri);
                        if (path == null) {
                            filename = FilenameUtils.getName(uri.toString());
                        } else {
                            File file = new File(path);
                            filename = file.getName();
                        }
                    } else {
                        Uri returnUri = data.getData();
                        Cursor returnCursor = getContentResolver().query(returnUri, null, null, null, null);
                        int nameIndex = returnCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME);
                        int sizeIndex = returnCursor.getColumnIndex(OpenableColumns.SIZE);
                        returnCursor.moveToFirst();
                        filename = returnCursor.getString(nameIndex);
                        String size = Long.toString(returnCursor.getLong(sizeIndex));
                    }
   File fileSave = getExternalFilesDir(null);
    String sourcePath = getExternalFilesDir(null).toString();
    try {
                        copyFileStream(new File(sourcePath + "/" + filename), uri,this);

                    } catch (Exception e) {
                        e.printStackTrace();
                    }
  }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}
private void copyFileStream(File dest, Uri uri, Context context)
        throws IOException {
    InputStream is = null;
    OutputStream os = null;
    try {
        is = context.getContentResolver().openInputStream(uri);
        os = new FileOutputStream(dest);
        byte[] buffer = new byte[1024];
        int length;

        while ((length = is.read(buffer)) > 0) {
            os.write(buffer, 0, length);

        }
    } catch (Exception e) {
        e.printStackTrace();
    } finally {
        is.close();
        os.close();
    }
}

在此之后,您可以通过适当的操作从保存文件的应用程序外部存储中打开此文件。

原文由 KJEjava48 发布,翻译遵循 CC BY-SA 4.0 许可协议

使用系统文件选择器选择任何文件:

 val intent = Intent(Intent.ACTION_GET_CONTENT)
intent.type = "*/*"
startActivityForResult(intent, 1)

活动结果:

 override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    super.onActivityResult(requestCode, resultCode, data)
    if (requestCode == 1 && resultCode == Activity.RESULT_OK) {
        data?.data?.let {
            getFileFromUri(requireContext().contentResolver, uri, requireContext().cacheDir)
        }
    }
}

获取文件:

 private fun getFileFromUri(contentResolver: ContentResolver, uri: Uri, directory: File): File {
    val file =
        File.createTempFile("suffix", "prefix", directory)
    file.outputStream().use {
        contentResolver.openInputStream(uri)?.copyTo(it)
    }

    return file
}

原文由 Abhi Muktheeswarar 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
logo
Stack Overflow 翻译
子站问答
访问
宣传栏