从URI获取真实路径,Android奇巧新的存储访问框架

Android 4.4(奇巧)的新画廊访问之前,我用这个方法在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

现在:

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

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

266767 次浏览

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

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

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

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

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

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

documentprovider使用示例

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

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

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

不要求存在与Uri对应的文件。

是的,我真的需要一条路

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

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

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

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

你可以使用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();

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

试试这个:

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

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

它为我工作,如果我把意图改为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 : "";
}

下面的答案是由https://stackoverflow.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;
}

编辑:代码上有一个流程;如果设备有多个外部存储(外部sd卡,外部usb等),以上代码将不能工作在非主存储。

这将从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

我们需要在之前的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());
}
}