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

本文介绍了解决在Android开发中遇到的文件查看器问题,即通过将content URI转换为真实文件路径来避免FileNotFoundException。提供了一个实用的ReaderUtils工具类,用于查询内容解析器并找到文件的真实存储地址。

问题:

最近在开发文件查看器中出现一个问题: 接收到一个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";

    /**
     * 查询内容解析器,找到文件存储地址
     * <p>ef: android中转换content://media/external/images/media/539163为/storage/emulated/0/DCIM/Camera/IMG_20160807_123123.jpg
     * <p>把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);
}

有疑问可以留言交流哦!

<think>首先,用户的问题是:"我想在UniApp中将content URI转换为本地文件路径 请问uniapp convert content uri to local file path"。这是一个关于UniApp开发的问题,具体是如何将content URI转换为本地文件路径。 在之前的引用中,有提到UniApp项目案例,比如引用[2]说:"博主作品:《Java项目案例》...《Uniapp项目案例》有几个有uniapp教程,企业实战开发..."。所以,我可以参考这个,但需要基于UniApp的知识来回答。 关键点是:在UniApp中,如何将content URI转换为本地文件路径Content URI通常是在Android系统中,用于访问内容提供者的URI,比如从图库选择图片后返回的URI。 在UniApp中,由于它是一个跨平台框架,使用Vue.js开发,运行在多个平台(如H5、App、小程序),所以处理方式可能因平台而异。用户可能是在App端(Android或iOS)遇到这个问题。 步骤: 1. **理解content URI**:在Android中,content URI(如content://media/external/images/media/123)指向文件系统中的一个文件,但需要通过内容解析器(ContentResolver)来访问。 2. **在UniApp中处理**:UniApp提供了uni.chooseImage等API来选择图片,但返回的可能是临时路径或URI。用户需要将URI转换为可用的本地文件路径。 在UniApp中,当使用uni.chooseImage选择图片时,它会返回一个临时路径(在H5中是base64或blob,在App中是临时文件路径)。但如果是通过其他方式获取的content URI,可能需要额外处理。 解决方案: - 在App端(Android),可以使用uni-app的Native.js或插件来调用原生方法。 - 或者使用uni-app的API直接处理。 一个常见的方法是使用uni.getFileInfo或uni.saveFile来保存文件并获取路径。 具体步骤: 1. 使用uni.chooseImage选择图片,它会返回临时路径。 2. 如果需要处理content URI,可能需要使用uni.downloadFile或uni.saveFile转换为本地路径。 对于content URI的转换,在UniApp中,可以通过以下方式: - 在Android上,使用plus.android.invoke来调用Java方法。 - 或者使用uni-app的扩展API。 一个标准的做法是: - 使用uni.getFileSystemManager()来管理文件系统。 - 或者使用uni.saveFile保存文件。 参考UniApp官方文档: - uni.chooseImage返回的tempFilePaths是临时路径,可以直接使用。 - 但如果是从intent获取的content URI,需要额外处理。 用户说"convert content uri to local file path",所以假设用户已经有一个content URI,需要转换。 在UniApp中,可以使用plus.io.resolveLocalFileSystemURL或类似方法。 具体代码示例: 在App端,可以使用plus.android.importClass来导入Android类,然后使用ContentResolver。 例如: ```javascript //Android平台 if (plus.os.name === 'Android') { const Context = plus.android.importClass('android.content.Context'); const ContentResolver = plus.android.importClass('android.content.ContentResolver'); const Uri = plus.android.importClass('android.net.Uri'); // 假设contentUri是字符串,如'content://media/external/images/media/123' const uri = Uri.parse(contentUri); const context = plus.android.runtimeMainActivity(); // 获取上下文 const contentResolver = context.getContentResolver(); // 打开输入流 const inputStream = contentResolver.openInputStream(uri); // 然后保存到本地文件 // 使用plus.io来保存 } ``` 但更简单的方法是使用uni-app的uni.downloadFile,但它不支持content URI直接下载。 另一个方法是使用uni.saveFile: ```javascript uni.saveFile({ tempFilePath: contentUri, // 这可能不工作,因为contentUri不是标准临时路径 success: function (res) { console.log('Saved file path: ' + res.savedFilePath); // 本地路径 } }); ``` 但uni.saveFile通常用于保存临时文件到本地。 对于content URI,最好使用原生方法。 一个推荐的做法是使用uni-app插件或自定义模块。 在UniApp社区中,有类似问题:使用plus.io.resolveLocalFileSystemURL来获取文件路径。 例如: ```javascript plus.io.resolveLocalFileSystemURL(contentUri, function(entry) { entry.file(function(file) { console.log('File path: ' + file.fullPath); // 但这可能不是绝对路径 //Android上,file.fullPath可能是虚拟路径 }); }); ``` 但在Android上,对于content URI,这可能不直接工作。需要将内容复制到应用私有目录。 标准做法是: 1. 使用ContentResolver打开输入流。 2. 读取数据。 3. 写入到应用缓存目录或文件目录。 然后获取本地路径。 代码示例: ```javascript //Android平台处理content URI function convertContentUriToLocalPath(contentUri, callback) { if (plus.os.name === 'Android') { const Context = plus.android.importClass('android.content.Context'); const ContentResolver = plus.android.importClass('android.content.ContentResolver'); const Uri = plus.android.importClass('android.net.Uri'); const File = plus.android.importClass('java.io.File'); const FileOutputStream = plus.android.importClass('java.io.FileOutputStream'); const InputStream = plus.android.importClass('java.io.InputStream'); const BufferedInputStream = plus.android.importClass('java.io.BufferedInputStream'); const BufferedOutputStream = plus.android.importClass('java.io.BufferedOutputStream'); const context = plus.android.runtimeMainActivity(); const resolver = context.getContentResolver(); const uri = Uri.parse(contentUri); let inputStream = null; try { inputStream = resolver.openInputStream(uri); // 创建输出文件在缓存目录 const cacheDir = context.getCacheDir(); const fileName = 'temp_' + new Date().getTime() + '.jpg'; // 假设是图片 const outputFile = new File(cacheDir, fileName); const outputStream = new FileOutputStream(outputFile); // 复制流 const buffer = plus.android.newArray('byte', 1024); let bytesRead; while ((bytesRead = inputStream.read(buffer)) !== -1) { outputStream.write(buffer, 0, bytesRead); } outputStream.flush(); outputStream.close(); inputStream.close(); // 获取绝对路径 const localPath = outputFile.getAbsolutePath(); callback(null, localPath); } catch (e) { callback(e, null); } } else { // 对于iOS或其他平台,处理不同 callback(new Error('Unsupported platform'), null); } } ``` 然后调用这个函数。 在UniApp中,更优雅的方式是使用uni.saveFile或uni.downloadFile,但它们可能不支持content URI。 对于iOS,content URI类似,但使用assets-library URL。 所以,需要平台特定代码。 在UniApp中,可以使用条件编译: ```javascript // #ifdef APP-PLUS // 平台特定代码 // #endif ``` 但用户可能希望一个简单的解决方案。 一个更简单的方法是:如果是从uni.chooseImage获取的,直接使用tempFilePaths。 如果用户是从其他来源获取content URI,则需要上述方法。 在回答中,我应该: - 解释问题。 - 提供解决方案。 - 给出代码示例。 - 确保可靠。 参考引用:用户提到了引用[2]关于UniApp项目,所以我可以在回答中引用,但引用[2]是描述性的,不是具体代码。所以,我基于知识回答。 生成相关问题:系统指令要求生成相关问题。 回答结构: 1. 简要介绍问题。 2. 提供详细步骤和代码。 3. 结束前添加相关问题。 使用中文回答。 最终回答:</think>在 UniApp 中,将 Content URI(如 Android 平台上的 `content://` 格式的 URI)转换为本地文件路径是一个常见需求,通常出现在文件选择(如图库选图)后需要处理真实文件路径的场景。以下是详细解决方案,基于 UniApp 的跨平台特性和原生扩展能力实现。方法分为两步:**获取 Content URI** 和**转换为本地路径**,核心是使用 UniApp 的 API 结合平台原生代码(Android/iOS)。 ### 步骤 1: 获取 Content URI 在 UniApp 中,使用 `uni.chooseImage` API 选择图片时,返回的临时路径(`tempFilePaths`)在 Android 上可能是 Content URI。示例: ```javascript uni.chooseImage({ count: 1, success: (res) => { const tempFilePaths = res.tempFilePaths; // 可能为 content:// URI this.convertToLocalPath(tempFilePaths[0]); // 调用转换函数 } }); ``` - **注意**:`tempFilePaths` 在 H5 平台是 base64 或 blob URL,在 App 平台(Android)可能直接是 Content URI。 ### 步骤 2: 转换为本地文件路径 由于 UniApp 未提供直接转换 Content URI 的 API,需调用原生能力。以下方法使用 `plus.android`(Android)和 `plus.ios`(iOS)实现跨平台兼容: #### 通用转换函数(支持 Android 和 iOS) ```javascript // #ifdef APP-PLUS function convertContentUriToLocalPath(contentUri, callback) { if (plus.os.name === 'Android') { // Android 平台处理 const Context = plus.android.importClass('android.content.Context'); const ContentResolver = plus.android.importClass('android.content.ContentResolver'); const Uri = plus.android.importClass('android.net.Uri'); const File = plus.android.importClass('java.io.File'); const FileOutputStream = plus.android.importClass('java.io.FileOutputStream'); const InputStream = plus.android.importClass('java.io.InputStream'); const context = plus.android.runtimeMainActivity(); const resolver = context.getContentResolver(); const uri = Uri.parse(contentUri); try { const inputStream = resolver.openInputStream(uri); const cacheDir = context.getCacheDir(); // 应用缓存目录 const fileName = `file_${Date.now()}.tmp`; // 生成唯一文件名 const outputFile = new File(cacheDir, fileName); const outputStream = new FileOutputStream(outputFile); // 复制数据流 const buffer = plus.android.newArray('byte', 1024); let bytesRead; while ((bytesRead = inputStream.read(buffer)) !== -1) { outputStream.write(buffer, 0, bytesRead); } // 关闭流并返回路径 outputStream.close(); inputStream.close(); const localPath = outputFile.getAbsolutePath(); // 如 /data/user/0/.../cache/file_123.tmp callback(null, localPath); } catch (e) { callback(e, null); } } else if (plus.os.name === 'iOS') { // iOS 平台处理(类似 PHAsset 转换) plus.ios.import('Photos').PHAsset.fetchAssetsWithALAssetURLsOptions( [contentUri], null, (assets) => { if (assets.count() > 0) { const asset = assets.firstObject(); const options = plus.ios.newObject('PHImageRequestOptions'); options.setValueForKey(true, 'synchronous'); plus.ios.import('PHImageManager').defaultManager().requestImageDataForAssetOptionsResultHandler( asset, options, (data, uti, orientation, info) => { const nsData = plus.ios.import('NSData').dataWithData(data); const fileManager = plus.ios.import('NSFileManager').defaultManager(); const cachePath = NSSearchPathForDirectoriesInDomains(plus.ios.import('NSCachesDirectory'), plus.ios.import('NSUserDomainMask'), true).firstObject(); const filePath = cachePath.stringByAppendingPathComponent(`file_${Date.now()}.tmp`); if (fileManager.createFileAtPathContentsAttributes(filePath, nsData, null)) { callback(null, filePath); // 如 /var/.../Caches/file_123.tmp } else { callback(new Error('File save failed'), null); } } ); } } ); } else { callback(new Error('Unsupported platform'), null); } } // #endif ``` #### 调用示例 ```javascript // 在 uni.chooseImage 的 success 回调中使用 convertContentUriToLocalPath(tempFilePaths[0], (err, localPath) => { if (err) { console.error('转换失败:', err); } else { console.log('本地路径:', localPath); // 输出如 /data/.../cache/file_123.tmp // 使用 localPath 进行文件上传或操作 uni.uploadFile({ url: 'https://example.com/upload', filePath: localPath, name: 'file' }); } }); ``` ### 关键说明 1. **平台兼容性**: - **Android**:通过 `ContentResolver` 读取 URI 数据,并保存到应用缓存目录(无需权限)。 - **iOS**:使用 `PHAsset` 处理相册资源,保存到 `NSCachesDirectory`。 - **H5/小程序**:无需转换,`tempFilePaths` 可直接使用(非 Content URI)。 2. **路径有效期**:转换后的文件保存在缓存目录,应用重启后可能被清除。如需持久化,使用 `uni.saveFile` 移动到用户目录。 3. **性能优化**:大文件操作建议在异步线程执行,避免阻塞 UI。 4. **安全提示**:直接操作原生代码需测试真机兼容性,建议封装为 UniApp 原生插件复用[^2]。 ### 相关问题 1. 如何在 UniApp 中处理 iOS 的相册权限问题? 2. UniApp 中如何实现文件上传的进度监控? 3. 如何将转换后的本地文件路径转换为 Base64 格式? > 引用说明: > [^1]: Content URI 常见于 Android 文件系统,如 `content://media/external/images/media/123`,需通过系统解析器访问。 > [^2]: UniApp 原生扩展能力可参考企业级实战案例,优化性能兼容性,详见项目经验总结。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值