从URI获取真实路径,Android KitKat新的存储访问框架[重复]

Posted

技术标签:

【中文标题】从URI获取真实路径,Android KitKat新的存储访问框架[重复]【英文标题】:Get real path from URI, Android KitKat new storage access framework [duplicate] 【发布时间】:2013-12-02 18:24:59 【问题描述】:

android 4.4 (KitKat) 中的新图库访问之前,我通过这种方法在 SD 卡上获得了我的真实路径:

public String getPath(Uri uri) 
   String[] projection =  MediaStore.Images.Media.DATA ;
   Cursor cursor = managedQuery(uri, projection, null, null, null);
   startManagingCursor(cursor);
   int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
   cursor.moveToFirst();
 return cursor.getString(column_index);

现在,Intent.ACTION_GET_CONTENT 返回不同的数据:

之前:

content://media/external/images/media/62

现在:

content://com.android.providers.media.documents/document/image:62

我怎样才能获得 SD 卡上的真实路径?

【问题讨论】:

我想你可以在这里找到解决方案***.com/questions/19834842/… 你真的需要路径吗?也许您只是想从 URI 中获取 Bitmap 对象?如果是这样,我有一个解决方案。 是的,我真的需要一个路径,我想从新的存储访问框架中获取它。我的临时解决方案是避免使用新的存储访问。我已将 Intent.ACTION_GET_CONTENT 更改为 Intent.ACTION_PICK @Alvaro,你最后做了什么 在 4.4.2 版本中无法从我的应用程序中选择谷歌驱动器中的项目。我完全被困住了?任何解决方案 【参考方案1】:

这将从 MediaProvider、DownloadsProvider 和 ExternalStorageProvider 获取文件路径,同时回退到您提到的非官方 ContentProvider 方法。

/**
 * Get a file path from a Uri. This will get the the path for Storage Access
 * Framework Documents, as well as the _data field for the MediaStore and
 * other file-based ContentProviders.
 *
 * @param context The context.
 * @param uri The Uri to query.
 * @author paulburke
 */
public static String getPath(final Context context, final Uri uri) 

    final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;

    // DocumentProvider
    if (isKitKat && 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 getDataColumn(context, uri, null, null);
    
    // File
    else if ("file".equalsIgnoreCase(uri.getScheme())) 
        return uri.getPath();
    

    return null;


/**
 * Get the value of the data column for this Uri. This is useful for
 * MediaStore Uris, and other file-based ContentProviders.
 *
 * @param context The context.
 * @param uri The Uri to query.
 * @param selection (Optional) Filter used in the query.
 * @param selectionArgs (Optional) Selection arguments used in the query.
 * @return The value of the _data column, which is typically a file path.
 */
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 column_index = cursor.getColumnIndexOrThrow(column);
            return cursor.getString(column_index);
        
     finally 
        if (cursor != null)
            cursor.close();
    
    return null;



/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is ExternalStorageProvider.
 */
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());

这些来自我的开源库aFileChooser。

【讨论】:

请务必在github.com/iPaulPro/aFileChooser/blob/master/aFileChooser/src/…查看这些方法的最新版本以进行更改 我似乎无法让它处理来自 Google Drive com.google.android.apps.docs.files 的文件 @PaulBurke - 谢谢你,但是“TODO 处理非主卷”是什么? @PaulBurke,如果您在大约 1 年前回答了这里提出的一些问题,那就太好了...我想使用您的库,但是有很多未解决的问题和最后一次提交 1一年前我的结论是它已经停产了。 这在数以亿计的 Android 设备(Android 4.4+ 附带的任何设备)上将毫无用处,其中Uri 指向可移动存储上的某些内容,因为您无法任意访问可移动存储.对于来自不支持_data 模式的其他ContentProvider 的任何Uri(例如FileProvider),它也会失败。 Uri 是一个不透明的句柄;照此处理,使用ContentResolver 获取Uri 标识的内容。【参考方案2】:

注意:这个答案解决了部分问题。如需完整的解决方案(以库的形式),请查看Paul Burke's answer。

您可以使用URI获取document id,然后查询MediaStore.Images.Media.EXTERNAL_CONTENT_URIMediaStore.Images.Media.INTERNAL_CONTENT_URI(视SD卡情况而定)。

获取文档ID:

// Will return "image:x*"
String wholeID = DocumentsContract.getDocumentId(uriThatYouCurrentlyHave);

// Split at colon, use second item in the array
String id = wholeID.split(":")[1];

String[] column =  MediaStore.Images.Media.DATA ;     

// where id is equal to             
String sel = MediaStore.Images.Media._ID + "=?";

Cursor cursor = getContentResolver().
                          query(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, 
                          column, sel, new String[] id , null);

String filePath = "";

int columnIndex = cursor.getColumnIndex(column[0]);

if (cursor.moveToFirst()) 
    filePath = cursor.getString(columnIndex);
   

cursor.close();

参考:我无法找到该解决方案的来源。我想请原始海报在这里投稿。今晚会看起来更多。

【讨论】:

@Magnus 感谢您没有尝试上面的代码,但仍然分享您的意见。 ...that has nothing to do with the ID in MediaStore. If you somehow find a matching ID in MediaStore it's **most certainly** not the corresponding image that DocumentsProvider returned in its URI。如果有任何官方文档支持我的回答,我会参考它。但是,我只能说,上面的代码在我测试的几次中产生了所需的输出。如果您在将其视为错误之前尝试过解决方案,那将是有礼貌的。 “每个存储后端通过使用唯一的 COLUMN_DOCUMENT_ID 引用它们来显示单个文件和目录。文档 ID 必须是唯一的,并且一旦发布就不会更改,因为它们用于在设备重新启动时进行持久 URI 授权。”取自 Storage access framework 这是您从 DocumentsContract.getDocumentId 获得的 ID,您可以看到它与 MediaStore 无关,无论我是否有礼貌 @Magnus 请查看MediaDocumentsProvider 的源代码。 queryDocument(...) 方法使用文档 ID(将 typeid 使用 String#substring(..) 拆分后)查询 MediaStore.Images.Media.EXTERNAL_CONTENT_URI。也许,只需阅读else if (TYPE_IMAGE.equals(ident.type)) 块。 是的,但是 DocumentsProvider 的整个想法是它抽象出底层数据,因此您的方法使用起来不安全,它可能是在线服务中的图像,因此不是文件 uri,因此该方法使用起来不安全。 @Pam 您拥有的 URI不是文档 URI - 文档 URI 以 document: 开头。您可以使用DocumentsContract.isDocumentUri(uri) 确认这一点,它基本上检查uri 是否具有前缀"document"。看来您已经有一个 file 计划 URI。获取文件路径,可以使用uri.getPath()【参考方案3】:

下面的答案是由https://***.com/users/3082682/cvizv 在一个不再存在的页面上写的,因为他没有足够的代表来回答问题,所以我将其发布。我没有学分。

public String getImagePath(Uri uri)
   Cursor cursor = getContentResolver().query(uri, null, null, null, null);
   cursor.moveToFirst();
   String document_id = cursor.getString(0);
   document_id = document_id.substring(document_id.lastIndexOf(":")+1);
   cursor.close();

   cursor = getContentResolver().query( 
   android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
   null, MediaStore.Images.Media._ID + " = ? ", new String[]document_id, null);
   cursor.moveToFirst();
   String path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
   cursor.close();

   return path;

编辑:代码上有流程;如果设备有多个外部存储(外部 sdcard、外部 USB 等),上面的代码将不适用于非主存储。

【讨论】:

这是从图库中获取图像路径的正确答案。非常感谢。我已经尝试了很多解决方案,但你的工作。早些时候,我的代码正在使用 managedQuery 工作,但一些我如何更改 api 级别和 managedQuery api 已被弃用。所以我必须更改代码。我搜索尝试解决方案的数量,但没有任何效果,您的解决方案可以完美运行。再次感谢。如果我可以放弃两点,我肯定可以,但系统不允许。 所有关于这个问题的读者,只要使用这个解决方案就完美了!!!这是我在 Android L 和 Android M 上发现的唯一解决方案。 它给出 CursorIndexOutOfBoundsException: Index 0 requested, with a size of 0 感谢真正有效的解决方案。在尝试了提供的许多其他解决方案之后(即使有更高的投票)。这是对我有用的。我想知道是否有一种适用于所有情况的通用解决方案,这样人们就不必依赖反复试验的方式。 @ClassA 可能返回的游标没有data 字段。也许最好打印该光标的内容。如果没有详细信息,我无法提供帮助,请为此提出一个新问题。【参考方案4】:

在 KitKat 中访问新画廊之前,我通过这种方法在 sdcard 中获得了我的真实路径

那从来都不可靠。您从ACTION_GET_CONTENTACTION_PICK 请求返回的Uri 不必由MediaStore 索引,甚至不必代表文件系统上的文件。例如,Uri 可以表示一个流,其中一个加密文件会在运行中为您解密。

我怎样才能设法获得 sdcard 中的真实路径?

不要求有Uri对应的文件。

是的,我真的需要一条路

然后将文件从流中复制到您自己的临时文件中,并使用它。更好的是,直接使用流,避免使用临时文件。

我已将 Intent.ACTION_GET_CONTENT 更改为 Intent.ACTION_PICK

这对你的情况没有帮助。没有要求ACTION_PICK 响应是针对在文件系统上有一个文件的Uri,您可以通过某种方式神奇地导出该文件。

【讨论】:

一般我们使用路径来获取图片的Exif标签,例如以确定是否需要旋转图像以进行显示。通过将流复制到新文件,exif标签不会丢失吗?我尝试通过 FileOutputStream 从 inputStream 写入位图,但无法恢复原始 exif 标签。 @ılǝ:“通过将流复制到新文件,exif 标签不会丢失吗?” ——他们不应该。但是,将图像解码为 Bitmap 会丢失 EXIF 标签。不过,只要您将其保留为 byte[],并且不尝试对这些字节进行任何修改,我希望 EXIF 标记能够继续存在。 @CommonsWare 好的,所以您建议使用 URI 而不是尝试获取文件路径。但是现在使用不同的 URI 格式将无法正常工作 Intent intent = new Intent(Intent.ACTION_VIEW); intent.setDataAndType(uri, "video/*"); startActivity(intent); 这个答案被低估了。它可能也应该提到 openFileDescriptor() 方法。也就是说,ParcelFileDescriptor 的存在一开始就没有得到很好的记录。 @SharpEdge:没有路径。您越早停止考虑文件和路径,而是考虑内容和Uri 值,您的生活就会越好。毕竟,图库的内容可能包括来自云存储提供商的文件(例如 Google 照片),以及您无法直接访问的文件(例如,可移动存储上的文件)。使用ContentResolverDocumentFile【参考方案5】:

我遇到了完全相同的问题。我需要文件名才能将其上传到网站。

如果我将意图更改为 PICK,它对我有用。 这在 AVD for Android 4.4 和 AVD for Android 2.1 中进行了测试。

添加权限 READ_EXTERNAL_STORAGE :

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

改变意图:

Intent i = new Intent(
  Intent.ACTION_PICK,
  android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI
  );
startActivityForResult(i, 66453666);

/* OLD CODE
Intent intent = new Intent();
intent.setType("image/*");
intent.setAction(Intent.ACTION_GET_CONTENT);
startActivityForResult(
  Intent.createChooser( intent, "Select Image" ),
  66453666
  );
*/

我无需更改代码即可获取实际路径:

// Convert the image URI to the direct file system path of the image file
 public String mf_szGetRealPathFromURI(final Context context, final Uri ac_Uri )
 
     String result = "";
     boolean isok = false;

     Cursor cursor = null;
      try  
        String[] proj =  MediaStore.Images.Media.DATA ;
        cursor = context.getContentResolver().query(ac_Uri,  proj, null, null, null);
        int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
        cursor.moveToFirst();
        result = cursor.getString(column_index);
        isok = true;
       finally 
        if (cursor != null) 
          cursor.close();
        
      

      return isok ? result : "";
 

【讨论】:

您的答案是我提到的临时解决方案。我想将意图与 GET_CONTENT 一起使用,并尝试充分利用新的选择器 ui。虽然不知道有没有可能。 这会引发“列'_data'不存在”异常。 @Jan Van Overbeke 我们是否需要在选择内容时指定任何 MIME 类型?不幸的是,当我在 6.0 上测试它时,它仍然打开 DocumentsProvider 而不是画廊选择器...:(为什么?【参考方案6】:

这个答案是基于你有点模糊的描述。我假设您通过操作触发了意图:Intent.ACTION_GET_CONTENT

现在您得到的是content://com.android.providers.media.documents/document/image:62,而不是之前的媒体提供者 URI,对吗?

在 Android 4.4 (KitKat) 上,新的 DocumentsActivity 在Intent.ACTION_GET_CONTENT 被触发时打开,从而导致您可以选择图像的网格视图(或列表视图),这将返回以下 URI 到调用上下文(示例) :content://com.android.providers.media.documents/document/image:62(这些是新文档提供者的 URI,它通过向客户端提供通用文档提供者 URI 来抽象出底层数据)。

但是,您可以使用 DocumentsActivity 中的抽屉访问库和响应 Intent.ACTION_GET_CONTENT 的其他活动(从左向右拖动,您将看到一个带有库的抽屉 UI 可供选择)。就像之前的 KitKat。

如果您仍然要在 DocumentsActivity 类中选择哪个并需要文件 URI,您应该能够执行以下操作(警告这是 hacky!)查询(使用 contentresolver):content://com.android.providers.media.documents/document/image:62 URI 并从光标。这是一个有点独特的名称(只是本地文件上的文件名),并在选择(查询时)中使用它向 mediaprovider 获取与此选择相对应的正确行,您也可以从此处获取文件 URI。

访问文档提供者的推荐方法可以在这里找到(获取输入流或文件描述符以读取文件/位图):

Examples of using documentprovider

【讨论】:

Google 刚刚更新了文档,太好了。对不起我的模糊描述。我想知道从新的选择器 ui 到离线过程的真实路径。我知道如何获取文件名但不知道完整路径,我想从这个数据内容中获取这个路径/storage/sdcard0/DCIM/Camera/IMG_20131118_153817_119.jpg//com.android.providers.media。文件/文件/图像:62 我认为你运气不好。如果您不愿意更改为使用 mime 类型 image/* 选择意图,也可以使用上面建议的方法(hack)。由于谷歌已经通过这个新的 DocumentActivity 获得了 Intent.ACTION_GET_CONTENT 的所有权,您唯一的选择是使用上面链接中描述的方法。他们似乎认为使用当前方法就足够了,因为文档提供者还包装了云端服务(例如 Drive 等),那么唯一的选择是从文档提供者返回输入流(在这种情况下没有文件)。 查看此链接:Columns in document provider 一旦您将光标访问到该文档提供者,您就可以使用这些链接,因此您没有文件 uri 选项 :(【参考方案7】:

试试这个:

//KITKAT
i = new Intent(Intent.ACTION_PICK,android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
startActivityForResult(i, CHOOSE_IMAGE_REQUEST);

在 onActivityResult 中使用以下内容:

Uri selectedImageURI = data.getData();
input = c.getContentResolver().openInputStream(selectedImageURI);
BitmapFactory.decodeStream(input , null, opts);

【讨论】:

仅适用于图像...【参考方案8】:

这是Paul Burke's answer 的更新版本。在Android 4.4 (KitKat) 以下的版本中,我们没有DocumentsContract 类。

为了在低于 KitKat 的版本上工作,请创建此类:

public class DocumentsContract 
    private static final String DOCUMENT_URIS =
        "com.android.providers.media.documents " +
        "com.android.externalstorage.documents " +
        "com.android.providers.downloads.documents " +
        "com.android.providers.media.documents";

    private static final String PATH_DOCUMENT = "document";
    private static final String TAG = DocumentsContract.class.getSimpleName();

    public static String getDocumentId(Uri documentUri) 
        final List<String> paths = documentUri.getPathSegments();
        if (paths.size() < 2) 
            throw new IllegalArgumentException("Not a document: " + documentUri);
        

        if (!PATH_DOCUMENT.equals(paths.get(0))) 
            throw new IllegalArgumentException("Not a document: " + documentUri);
        
        return paths.get(1);
    

    public static boolean isDocumentUri(Uri uri) 
        final List<String> paths = uri.getPathSegments();
        Logger.v(TAG, "paths[" + paths + "]");
        if (paths.size() < 2) 
            return false;
        
        if (!PATH_DOCUMENT.equals(paths.get(0))) 
            return false;
        
        return DOCUMENT_URIS.contains(uri.getAuthority());
    

【讨论】:

【参考方案9】:

我们需要在我们之前的 onActivityResult() 的画廊选择器代码中进行以下更改/修复,以便在 Android 4.4 (KitKat) 和所有其他早期版本上无缝运行。

Uri selectedImgFileUri = data.getData();

if (selectedImgFileUri == null ) 

    // The user has not selected any photo


try 

   InputStream input = mActivity.getContentResolver().openInputStream(selectedImgFileUri);
   mSelectedPhotoBmp = BitmapFactory.decodeStream(input);

catch (Throwable tr) 

    // Show message to try again

【讨论】:

以上是关于从URI获取真实路径,Android KitKat新的存储访问框架[重复]的主要内容,如果未能解决你的问题,请参考以下文章

android 通过 Uri.getPath() 获取真实路径

java 从URI获取真实路径。字体:https://stackoverflow.com/questions/2789276/android-get-real-path-by-uri-getpath

获取 Uri Android 的真实路径

[Android - Kitkat ]以编程方式从 Android 的内置图库应用程序中获取/选择图像

java 从URI中获取图片真实路径

Android从Uri获取视频图片的真实地址