Android开发中content://与file://真实文件路径的转换

您所在的位置:网站首页 sdcard0找不到 Android开发中content://与file://真实文件路径的转换

Android开发中content://与file://真实文件路径的转换

2023-09-30 10:28| 来源: 网络整理| 查看: 265

问题:

最近在开发文件查看器中出现一个问题: 接收到一个uri: content://media/external/images/media/2283  获取到的getPath: /external/images/media/2283, 打开文件是出现异常:

java.io.FileNotFoundException: open failed: ENOENT (No such file or directory) at android.database.DatabaseUtils.readExceptionWithFileNotFoundExceptionFromParcel(DatabaseUtils.java:144) at android.content.ContentProviderProxy.openTypedAssetFile(ContentProviderNative.java:698) at android.content.ContentResolver.openTypedAssetFileDescriptor(ContentResolver.java:1416) at android.content.ContentResolver.openAssetFileDescriptor(ContentResolver.java:1253) at android.content.ContentResolver.openInputStream(ContentResolver.java:973)  

解决方案:

设法把content uri转化为真是文件路径,即可使用FIleInputStream获取输入流打开文件,代码如下:

ReaderUtils.java工具类:

package com.hulk.app.common.utils; import android.content.Context; import android.database.Cursor; import android.net.Uri; import android.provider.MediaStore; import android.util.Log; import java.io.File; /** * 文件阅读器工具类 * @author zhanghao */ public class ReaderUtils { public static final String TAG = "ReaderUtils"; /** * 查询内容解析器,找到文件存储地址 *

ef: android中转换content://media/external/images/media/539163为/storage/emulated/0/DCIM/Camera/IMG_20160807_123123.jpg *

把content://media/external/images/media/X转换为file:///storage/sdcard0/Pictures/X.jpg * @param context * @param contentUri * @return */ public static String getRealPathFromUri(Context context, Uri contentUri) { Cursor cursor = null; try { Log.i(TAG, "getRealPathFromUri: " + contentUri); String[] proj = { MediaStore.Images.Media.DATA }; cursor = context.getContentResolver().query(contentUri, proj, null, null, null); if (cursor != null && cursor.getColumnCount() > 0) { cursor.moveToFirst(); int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA); String path = cursor.getString(column_index); Log.i(TAG, "getRealPathFromUri: column_index=" + column_index + ", path=" + path); return path; } else { Log.w(TAG, "getRealPathFromUri: invalid cursor=" + cursor + ", contentUri=" + contentUri); } } catch (Exception e) { Log.e(TAG, "getRealPathFromUri failed: " + e + ", contentUri=" + contentUri, e); } finally { if (cursor != null) { cursor.close(); } } return ""; } /** * 获取完整文件名(包含扩展名) * @param filePath * @return */ public static String getFilenameWithExtension(String filePath) { if (filePath == null || filePath.length() == 0) { return ""; } int lastIndex = filePath.lastIndexOf(File.separator); String filename = filePath.substring(lastIndex + 1); return filename; } /** * 判断文件路径的文件名是否存在文件扩展名 eg: /external/images/media/2283 * @param filePath * @return */ public static boolean isFilePathWithExtension(String filePath) { String filename = getFilenameWithExtension(filePath); return filename.contains("."); } }

具体用法举例:

documentType = DocumentType.Type.Text;         try{             mFilePath = uri.getPath();             //问题: 当uri为 content://media/external/images/media/2283 时, path为 /external/images/media/2283             // 此时getOfficeTypeByFileName函数无法扥根据后缀名获取文件类型,后面的加载文件直接报错误             //解决方案,根据url的path判断文件是否存在后缀名,如果不存在后缀名需要从系统数据库中获取真正的文件路径             isFilePathWithExtension = ReaderUtils.isFilePathWithExtension(mFilePath);             if (!isFilePathWithExtension) {                 String path = ReaderUtils.getRealPathFromUri(getApplicationContext(), uri);                 Log.w(TAG, "onCreate: Got real file path " + path + " from uri " + uri);                 if (!TextUtils.isEmpty(path)) {                     mFilePath = path;                 }             }             documentType = DocumentUtils.getOfficeTypeByFileName(mFilePath);             fileName = getFileName(mFilePath);         }catch (Exception e){             Log.e(TAG, "Get file info error:" + e + ",  uri: " + uri + ", url path: " + mFilePath, e);         }

此处省略无数行......................

try { Log.i(TAG, "Start loadDocument uri:" + uri + ", documentType:" + documentType); IDocumentController mDocumentsContract = mDocumentView.getDocumentController(this); //mDocumentsContract.loadDocument(getContentResolver().openInputStream(uri), documentType, ENABLE_COPY); if (!isFilePathWithExtension && !TextUtils.isEmpty(mFilePath)) { //java.io.FileNotFoundException: open failed: ENOENT (No such file or directory), // uri:content://media/external/images/media/2283, documentType:0 //此处优先使用文件输入流加载,避免出现上述异常 Log.w(TAG, "onCreate: Load by FileInputStream from file path: " + mFilePath); FileInputStream in = new FileInputStream(mFilePath); if (in != null) { Log.w(TAG, "onCreate: loadDocument InputStream available: " + in.available()); mDocumentsContract.loadDocument(in, documentType, ENABLE_COPY); } else { mDocumentsContract.loadDocument(getContentResolver().openInputStream(uri), documentType, ENABLE_COPY); } } else { mDocumentsContract.loadDocument(getContentResolver().openInputStream(uri), documentType, ENABLE_COPY); } } catch (Exception e) { Log.e(TAG, "loadDocument error:" + e + ", uri:" + uri + ", documentType:" + documentType, e); }

有疑问可以留言交流哦!



【本文地址】


今日新闻


推荐新闻


CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3