在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);
}

现在来看意图。ACTION_GET_CONTENT返回不同的数据:

之前:

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

Now:

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

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


当前回答

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

这从来都不可靠。不要求从ACTION_GET_CONTENT或ACTION_PICK请求返回的Uri必须由MediaStore索引,甚至必须表示文件系统上的一个文件。例如,Uri可以表示一个流,其中加密的文件会为您动态解密。

我怎么能设法获得真正的路径在sdcard?

不需要有一个与Uri对应的文件。

是的,我真的需要一条路

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

我已经改变了我的意图。ACTION_GET_CONTENT for Intent。ACTION_PICK

这对你的处境没有帮助。没有要求ACTION_PICK响应是针对文件系统上有一个可以神奇地派生的文件的Uri。

其他回答

这个答案是基于你有些模糊的描述。我假设你用行动触发了一个意图:意图。ACTION_GET_CONTENT

现在你得到内容://com.android。providers。media。documents/document/image:62返回,而不是之前的媒体提供者URI,对吗?

在Android 4.4 (KitKat)上,当一个Intent被打开时,新的DocumentsActivity被打开。ACTION_GET_CONTENT被触发,从而导致网格视图(或列表视图),在那里你可以选择一个图像,这将返回以下uri调用context(示例):documents/document/image:62(这些是新文档提供程序的uri,它通过向客户端提供通用文档提供程序uri来抽象底层数据)。

然而,您可以访问gallery和其他响应Intent的活动。通过使用DocumentsActivity中的抽屉来ACTION_GET_CONTENT(从左到右拖动,您将看到一个有Gallery可供选择的抽屉UI)。就像奇巧之前一样。

If you still which to pick in DocumentsActivity class and need the file URI, you should be able to do the following (warning this is hacky!) query (with contentresolver):content://com.android.providers.media.documents/document/image:62 URI and read the _display_name value from the cursor. This is somewhat unique name (just the filename on local files) and use that in a selection (when querying) to mediaprovider to get the correct row corresponding to this selection from here you can fetch the file URI as well.

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

使用documentprovider的例子

我也有同样的问题。我需要的文件名,以便能够上传到一个网站。

如果我将意图改为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 : "";
 }

我们需要在我们早期的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
}

以下是保罗·伯克的回答的更新版本。在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());
    }
}

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

这从来都不可靠。不要求从ACTION_GET_CONTENT或ACTION_PICK请求返回的Uri必须由MediaStore索引,甚至必须表示文件系统上的一个文件。例如,Uri可以表示一个流,其中加密的文件会为您动态解密。

我怎么能设法获得真正的路径在sdcard?

不需要有一个与Uri对应的文件。

是的,我真的需要一条路

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

我已经改变了我的意图。ACTION_GET_CONTENT for Intent。ACTION_PICK

这对你的处境没有帮助。没有要求ACTION_PICK响应是针对文件系统上有一个可以神奇地派生的文件的Uri。