SAF文件选择与真实路径转换

一、构建选择文件与回调方法

//文件选择回调ActivityResultLauncher<String[]> pickFile = registerForActivityResult(new ActivityResultContracts.OpenDocument(), uri->{if (uri != null) {Log.e("cxy", "返回的uri:" + uri);Log.e("cxy","Path是:"+ FileUtil.getFilePathByUri(context,uri));}});//启动选择
pickFile.launch(new String[]{"*/*"})

二、uri转真实路径工具类

备注:针对Download目录先判空,为空则复制文件至内部路径再返回

public class FileUtil {public static String getFilePathByUri(Context context, Uri uri) {// 以 file:// 开头的if (ContentResolver.SCHEME_FILE.equals(uri.getScheme())) {return uri.getPath();}// 以/storage开头的也直接返回if (isOtherDocument(uri)) {return uri.getPath();}// 版本兼容的获取!String path = getFilePathByUri_BELOWAPI11(context, uri);if (path != null) {Log.e("cxy", "getFilePathByUri_BELOWAPI11获取到的路径为:" + path);return path;}path = getFilePathByUri_API11to18(context, uri);if (path != null) {Log.e("cxy", "getFilePathByUri_API11to18获取到的路径为:" + path);return path;}path = getFilePathByUri_API19(context, uri);Log.e("cxy", "getFilePathByUri_API19获取到的路径为:" + path);return path;}private static String getFilePathByUri_BELOWAPI11(Context context, Uri uri) {// 以 content:// 开头的,比如 content://media/extenral/images/media/17766if (ContentResolver.SCHEME_CONTENT.equals(uri.getScheme())) {String path = null;String[] projection = new String[]{MediaStore.Images.Media.DATA};Cursor cursor = context.getContentResolver().query(uri, projection, null, null, null);if (cursor != null) {if (cursor.moveToFirst()) {int columnIndex = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);if (columnIndex > -1) {path = cursor.getString(columnIndex);}}cursor.close();}return path;}return null;}private static String getFilePathByUri_API11to18(Context context, Uri contentUri) {String[] projection = {MediaStore.Images.Media.DATA};String result = null;CursorLoader cursorLoader = new CursorLoader(context, contentUri, projection, null, null, null);Cursor cursor = cursorLoader.loadInBackground();if (cursor != null) {int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);cursor.moveToFirst();result = cursor.getString(column_index);cursor.close();}return result;}private static String getFilePathByUri_API19(Context context, Uri uri) {// 4.4及之后的 是以 content:// 开头的,比如 content://com.android.providers.media.documents/document/image%3A235700if (ContentResolver.SCHEME_CONTENT.equals(uri.getScheme()) && Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {if (DocumentsContract.isDocumentUri(context, uri)) {if (isExternalStorageDocument(uri)) {// ExternalStorageProviderString docId = DocumentsContract.getDocumentId(uri);String[] split = docId.split(":");String type = split[0];if ("primary".equalsIgnoreCase(type)) {if (split.length > 1) {return Environment.getExternalStorageDirectory() + "/" + split[1];} else {return Environment.getExternalStorageDirectory() + "/";}// This is for checking SD Card}} else if (isDownloadsDocument(uri)) {//下载内容提供者时应当判断下载管理器是否被禁用int stateCode = context.getPackageManager().getApplicationEnabledSetting("com.android.providers.downloads");if (stateCode != 0 && stateCode != 1) {return null;}String id = DocumentsContract.getDocumentId(uri);// 如果出现这个RAW地址,我们则可以直接返回!if (id.startsWith("raw:")) {return id.replaceFirst("raw:", "");}if (id.contains(":")) {String[] tmp = id.split(":");if (tmp.length > 1) {id = tmp[1];}}Uri contentUri = Uri.parse("content://downloads/public_downloads");Log.e("cxy", "测试打印Uri: " + uri);try {contentUri = ContentUris.withAppendedId(contentUri, Long.parseLong(id));} catch (Exception e) {e.printStackTrace();}String path = getDataColumn(context, contentUri, null, null);if (path != null) return path;// 兼容某些特殊情况下的文件管理器!String fileName = getFileNameByUri(context, uri);if (fileName != null) {path = Environment.getExternalStorageDirectory().toString() + "/Download/" + fileName;File pathFile = new File(path);if (!pathFile.exists()){path = getFilePathForCopy(context,uri);}return path;}} else if (isMediaDocument(uri)) {// MediaProviderString docId = DocumentsContract.getDocumentId(uri);String[] split = docId.split(":");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;}String selection = "_id=?";String[] selectionArgs = new String[]{split[1]};return getDataColumn(context, contentUri, selection, selectionArgs);}}}return null;}private static String getFileNameByUri(Context context, Uri uri) {String relativePath = getFileRelativePathByUri_API18(context, uri);if (relativePath == null) relativePath = "";final String[] projection = {MediaStore.MediaColumns.DISPLAY_NAME};try (Cursor cursor = context.getContentResolver().query(uri, projection, null, null, null)) {if (cursor != null && cursor.moveToFirst()) {int index = cursor.getColumnIndexOrThrow(MediaStore.MediaColumns.DISPLAY_NAME);return relativePath + cursor.getString(index);}}return null;}private static String getFileRelativePathByUri_API18(Context context, Uri uri) {final String[] projection;if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.Q) {projection = new String[]{MediaStore.MediaColumns.RELATIVE_PATH};try (Cursor cursor = context.getContentResolver().query(uri, projection, null, null, null)) {if (cursor != null && cursor.moveToFirst()) {int index = cursor.getColumnIndexOrThrow(MediaStore.MediaColumns.RELATIVE_PATH);return cursor.getString(index);}}}return null;}private static String getFilePathForCopy(Context context, Uri uri) {try {Cursor returnCursor = context.getContentResolver().query(uri, null, null, null, null);int nameIndex = returnCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME);returnCursor.moveToFirst();String name = (returnCursor.getString(nameIndex));File file = new File(context.getFilesDir(), name);InputStream inputStream = context.getContentResolver().openInputStream(uri);FileOutputStream outputStream = new FileOutputStream(file);int read = 0;int maxBufferSize = 1024 * 1024;int bytesAvailable = inputStream.available();int bufferSize = Math.min(bytesAvailable, maxBufferSize);final byte[] buffers = new byte[bufferSize];while ((read = inputStream.read(buffers)) != -1) {outputStream.write(buffers, 0, read);}returnCursor.close();inputStream.close();outputStream.close();return file.getPath();} catch (Exception e) {e.printStackTrace();}return null;}private static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {final String column = MediaStore.Images.Media.DATA;final String[] projection = {column};try (Cursor 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);}} catch (IllegalArgumentException iae) {iae.printStackTrace();}return null;}private static boolean isExternalStorageDocument(Uri uri) {return "com.android.externalstorage.documents".equals(uri.getAuthority());}private static boolean isOtherDocument(Uri uri) {// 以/storage开头的也直接返回if (uri != null && uri.getPath() != null) {String path = uri.getPath();if (path.startsWith("/storage")) {return true;}if (path.startsWith("/external_files")) {return true;}}return false;}private static boolean isDownloadsDocument(Uri uri) {return "com.android.providers.downloads.documents".equals(uri.getAuthority());}private static boolean isMediaDocument(Uri uri) {return "com.android.providers.media.documents".equals(uri.getAuthority());}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/8695.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【JavaEE初阶系列】——利用Servlet API实现表白墙网站(1.6w字深刻了解前后端交互逻辑)

目录 &#x1f393;准备工作 &#x1f393;了解前端部分知识(主要JS&#xff09; &#x1f393;前后端交互部分 &#x1f388;主要思路 &#x1f6a9;提交信息 &#x1f36d;让前端发起一个ajax请求 &#x1f36d;服务器读取上述请求&#xff0c;并计算出响应 &#x1…

【Stylus详解与引入】

文章目录 Stylus详解与引入一、Stylus简介二、Stylus的特性1. 变量2. 嵌套规则3. 混合&#xff08;Mixins&#xff09;4. 函数5. 条件语句和循环 三、Stylus的引入与配置1. 安装Stylus和stylus-loader2. 配置Webpack3. 在Vue项目中使用Stylus4. 编译Stylus代码四、Stylus的性能…

Gateway的基本配置

Gateway的基本配置通常涉及网络、路由、安全性以及其他一些关键元素。以下是一个关于Gateway基本配置的详细指南&#xff1a; 网络连接配置&#xff1a; 设备连接&#xff1a;确保网关设备与内部网络的交换机、集线器或服务器/计算机正确连接。网络拓扑结构应支持内外通信。IP地…

2024-05-08 问AI: 在深度学习中,介绍一下RMSProp 优化器

文心一言 在深度学习中&#xff0c;RMSProp&#xff08;Root Mean Square Propagation&#xff09;优化器是一种常用的优化算法&#xff0c;主要用于神经网络训练的梯度下降算法的变体。它是对Adagrad优化器的一种改进&#xff0c;旨在解决Adagrad中学习率过快下降的问题。 R…

设计模式有哪些基本原则

目录 开闭原则(Open Closed Principle) 里氏替换原则(Liskov Substitution principle) 单一职责原则(Single Responsibility Principle,SRP)

RAC GCS_SERVER_PROCESSES参数

参考文档&#xff1a; GCS_SERVER_PROCESSES (oracle.com) 在awr报告中&#xff0c;看到addm建议如下&#xff1a; 其中有Global Cache Messaging和 Global Cache Congestion Activity During the Analysis Period ----------------------------------- Total database tim…

Linux的编译器

程序编译的过程 程序的编译过程是将源代码转换为可执行文件的一系列步骤。这个过程涉及多个阶段&#xff0c;主要包括预处理、编译、汇编和链接。下面详细介绍每个阶段&#xff1a; 1. 预处理&#xff08;Preprocessing&#xff09; 在实际编译之前&#xff0c;源代码文件首…

Android bootchart 分析启动性能工具使用

bootchart简介 bootchart 可为整个系统提供所有进程的 CPU 和 I/O 负载细分。该工具不需要重建系统映像&#xff0c;可以用作进入 systrace 之前的快速健全性检查。 1. 板端配置&#xff1a; 在Android 板端启用 bootchart&#xff0c;请运行以下命令&#xff1a; 2. Bootch…

Git克隆仓库报错:HTTP/2 stream 1 was not closed

报错及原因 fatal: unable to access ‘https://github.com/xxx/’: HTTP/2 stream 1 was not closed cleanly before end of the underlying stream http/2 和 http/1.1之间有个区别是“HTTP2 基于 SPDY&#xff0c;专注于性能&#xff0c;最大的一个目标是在用户和网站间只…

PMP考试未通过该怎么办?如何补考?

2024年3月10日PMP考试成绩正在陆续分批次发布。没有考试通过的同学就会疑问&#xff0c;考试没考过怎么办&#xff1f;可不可以补考&#xff1f;面对PMP考试没通过的情况&#xff0c;我们应该如何应对呢&#xff1f; PMP考试没合格可以补考&#xff0c;而且一年内有两次补考机…

编译官方原版的openwrt并加入第三方软件包

最近又重新编译了最新的官方原版openwrt-2305&#xff08;2024.3.22&#xff09;&#xff0c;此处记录一下以待日后参考。 目录 1.源码下载 1.1 通过官网直接下载 1.2 映射github加速下载 1.2.1 使用github账号fork源码 1.2.2 创建gitee账号映射github openwrt 2.编译准…

ChatGPT DALL-E绘图,制作各种表情包,实现穿衣风格的自由切换

DALL-E绘图功能探索&#xff1a; 1、保持人物形象一致&#xff0c;适配更多的表情、动作 2、改变穿衣风格 3、小女孩的不同年龄段展示 4、不同社交平台的个性头像创作 如果不会写代码&#xff0c;可以问GPT。使用地址&#xff1a;我的GPT4 视频&#xff0c;B站会发&#…

【YOLO改进】换遍MMDET主干网络之MobileNetv2(基于MMYOLO)

MobileNetv2 MobileNetV2是一种专为移动和嵌入式视觉应用设计的轻量化网络结构。它在MobileNetV1的基础上引入了两个主要的创新点&#xff1a;反转残差结构&#xff08;Inverted Residuals&#xff09;和线性瓶颈层&#xff08;Linear Bottlenecks&#xff09;。 反转残差结构…

Linux fsck.ext2命令教程:如何检查和修复ext2文件系统(附实例详解和注意事项)

Linux fsck.ext2命令介绍 fsck.ext2是一个用于检查和&#xff08;可选&#xff09;修复Linux ext2/ext3/ext4文件系统的命令。当ext2文件系统发生错误时&#xff0c;可以使用fsck.ext2命令尝试进行修复。 Linux fsck.ext2命令适用的Linux版本 fsck.ext2命令在大多数Linux发行…

yolov8任务之目标检测

对象检测 对象检测是一项涉及识别图像或视频流中对象的位置和类别的任务。对象检测器的输出是一组包围图像中对象的边界框&#xff0c;以及每个框的类标签和置信度分数。当您需要识别场景中感兴趣的对象&#xff0c;但不需要确切知道对象在哪里或其确切形状时&#xff0c;对象检…

我独自升级崛起账号注册 我独自升级怎么注册账号

近期&#xff0c;《我独自升级》这部动画凭借爆棚的人气&#xff0c;在各大平台上掀起了一阵观看热潮&#xff0c;其影响力不容小觑。借此时机&#xff0c;韩国游戏巨头网石集团敏捷响应&#xff0c;顺势推出了同名游戏《我独自升级&#xff1a;ARISE》&#xff0c;为粉丝们搭建…

SLAM种的回环检测

SLAM&#xff0c;即同步定位与建图&#xff0c;是一种在未知环境中&#xff0c;通过传感器&#xff08;如相机、激光雷达等&#xff09;进行自身定位与环境地图构建的技术。其中&#xff0c;回环检测是SLAM中的一个重要环节。 回环检测的主要目的是识别机器人是否回到了之前访…

webpack配置、插件使用案例

概念 本质上&#xff0c;webpack 是一个用于现代 JavaScript 应用程序的 静态模块打包工具。当 webpack 处理应用程序时&#xff0c;它会在内部从一个或多个入口点构建一个 依赖图(dependency graph)&#xff0c;然后将你项目中所需的每一个模块组合成一个或多个 bundles&…

关闭vscode保存自动格式化的功能

1 首先打开设置 搜索&#xff1a;editor.formatOnSave 取消勾选框 2 再打开 settings.json 搜索 editor 找到 settings.json 设置&#xff1a; "editor.formatOnSave": false

Java并发控制:深入解析Java BlockingQueue的用法与实践

1.BlockingQueue概述 1.1 BlockingQueue的定义与用途 BlockingQueue是一个支持两个附加操作的Queue&#xff0c;即在队列为空时获取元素的线程会等待队列变为非空&#xff0c;当队列满时存储元素的线程会等待队列可用。这主要用于生产者-消费者场景&#xff0c;其中生产者不能…