在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卡上的真实路径?


当前回答

这个答案是基于你有些模糊的描述。我假设你用行动触发了一个意图:意图。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的例子

其他回答

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

试试这个:

//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);

注意:这个答案解决了部分问题。要获得完整的解决方案(以库的形式),请参阅Paul Burke的答案。

您可以使用URI获取文档id,然后查询MediaStore.Images.Media。EXTERNAL_CONTENT_URI或MediaStore.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();

参考:我找不到这个解决方案的帖子。我想请原海报在这里投稿。今晚会再看看。

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

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

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

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

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

是的,我真的需要一条路

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

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

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