android把U盘的uri转为路径

时间: 2023-10-25 16:06:38 浏览: 174

在 Android 11 中,访问外部存储需要申请 MANAGE_EXTERNAL_STORAGE 权限。如果没有申请该权限,会导致访问外部存储时出现错误。

可以参考以下代码,在 AndroidManifest.xml 文件中申请该权限:

<uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" />

同时,在运行时也需要动态申请该权限。

以下是将 U 盘的 Uri 转换为路径的示例代码:

public class FileUtils {
    public static String getUsbDiskPath(Context context) {
        String usbDiskPath = null;
        File[] dirs = ContextCompat.getExternalFilesDirs(context, null);
        for (File dir : dirs) {
            if (dir != null &amp;&amp; dir.exists() &amp;&amp; dir.isDirectory() &amp;&amp; dir.canWrite()) {
                String dirPath = dir.getAbsolutePath();
                if (dirPath.startsWith("/storage/")) {
                    // 内置存储
                    continue;
                } else {
                    // 外置存储
                    usbDiskPath = dirPath;
                    break;
                }
            }
        }
        return usbDiskPath;
    }

    public static String getRealPathFromUri(Context context, Uri uri) {
        String realPath = null;
        if (DocumentsContract.isDocumentUri(context, uri)) {
            if (isExternalStorageDocument(uri)) {
                // 外部存储
                String docId = DocumentsContract.getDocumentId(uri);
                String[] split = docId.split(":");
                String type = split[0];
                if ("primary".equalsIgnoreCase(type)) {
                    realPath = Environment.getExternalStorageDirectory() + "/" + split[1];
                } else {
                    String usbDiskPath = getUsbDiskPath(context);
                    if (usbDiskPath != null) {
                        realPath = usbDiskPath + "/" + split[1];
                    }
                }
            } else if (isDownloadsDocument(uri)) {
                // 下载目录
                String id = DocumentsContract.getDocumentId(uri);
                Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));
                realPath = getDataColumn(context, contentUri, null, null);
            } else if (isMediaDocument(uri)) {
                // 媒体库
                String 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]};
                realPath = getDataColumn(context, contentUri, selection, selectionArgs);
            }
        } else if (ContentResolver.SCHEME_FILE.equalsIgnoreCase(uri.getScheme())) {
            // 直接从文件中获取
            realPath = uri.getPath();
        }
        return realPath;
    }

    private static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {
        Cursor cursor = null;
        String column = "_data";
        String[] projection = {column};
        try {
            cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null);
            if (cursor != null &amp;&amp; cursor.moveToFirst()) {
                int index = cursor.getColumnIndexOrThrow(column);
                return cursor.getString(index);
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (cursor != null) {
                cursor.close();
            }
        }
        return null;
    }

    private static boolean isExternalStorageDocument(Uri uri) {
        return "com.android.externalstorage.documents".equals(uri.getAuthority());
    }

    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());
    }
}

该工具类中的 getRealPathFromUri 方法可以将 U 盘的 Uri 转换为路径。具体实现是根据 Uri 的不同类型,采用不同的方式获取路径。

向AI提问 loading 发送消息图标

相关推荐

最新推荐

recommend-type

Android7.0以上Uri转路径的方法实现(已验证)

在Android 7.0及以上版本,由于引入了更好的文件安全性和隐私保护,传统的通过URI获取文件路径的方式不再适用。在这些版本中,Android系统限制了直接访问存储设备上的文件路径,而是鼓励使用ContentResolver和...
recommend-type

android图片文件的路径地址与Uri的相互转换方法

Android 图片文件的路径地址与 Uri 的相互转换方法 Android 开发中常遇到的问题之一是图片文件的路径地址与 Uri 之间的相互转换。这个问题可能会让许多开发者感到困惑和不知所措。下面我们将详细介绍如何将图片文件...
recommend-type

适配android7.0获取文件的Uri的方法

Android 7.0 获取文件的 Uri 的方法 ...适配 Android 7.0 获取文件的 Uri 的方法就是注册 ContentProvider,设置分享的路径,修改 java 代码来获取加密后的 Uri。这样可以确保文件的安全,避免了文件被恶意访问。
recommend-type

android获取相册图片和路径的实现方法

在Android应用开发中,获取用户从相册选择的图片及其路径是常见的需求。尤其是在Android 4.4(KitKat)及更高版本,由于系统API的变化,开发者需要采用新的方法来处理这一任务。以下是一个详细解释如何在Android 4.4...
recommend-type

详解android 通过uri获取bitmap图片并压缩

android 通过uri获取bitmap图片并压缩 在 Android 开发中,获取并压缩图片是一项非常重要的任务。特别是在调用图库选择图片时,需要通过uri获取bitmap图片并进行压缩,以避免OutOfMemoryError(OOM)。本文将详细...
recommend-type

Android中Comparable和Comparator的实现与应用

在学习和使用Java编程时,了解和掌握集合排序是十分重要的。在Java中,集合排序通常通过两种接口实现,即Comparable和Comparator。本文将通过一个名为"Android ComparableDemo"的实例程序,详细解析这两种排序接口的用途、区别和使用方法。 首先,我们需要明确Comparable接口的作用。Comparable接口位于java.lang包中,是一个泛型接口,它定义了一个单一的方法compareTo()。实现此接口的对象在进行排序时,会按照compareTo()方法所定义的顺序进行比较。换句话说, Comparable接口允许类进行自然排序,即在对象所属类的内部定义元素的排序规则。比如,如果我们有一个学生类,我们希望按照学生的分数进行排序,我们就会在学生类中实现Comparable接口,并重写compareTo()方法,以分数的高低作为排序标准。 而Comparator接口位于java.util包中,与Comparable不同,它是一个单独的类,不是定义在被排序对象的类中,而是定义在外部。Comparator提供了一个compare()方法,当需要进行比较的两个对象不具有相同的类,或者你希望使用不同的排序规则时,就会使用到Comparator。通过Comparator接口,可以在不修改对象类定义的情况下,对其进行排序。这种灵活性让Comparator非常适合于那些需要多种排序规则的场景,比如根据不同的属性来排序同一个对象列表。 在Android的开发中,Comparable和Comparator也被广泛用于列表和数组的排序。ComparableDemo示例程序便是用来演示如何使用Comparable接口来实现对象的自然排序。在这个程序中,很可能定义了一个类,比如Student,并且该类实现了Comparable接口。在这个例子中,Student类中的compareTo()方法会根据学生成绩或者其他属性来决定对象间的排序关系。 下面,我们将通过ComparableDemo的源代码,深入理解Comparable接口的实现和应用: ```java public class Student implements Comparable<Student> { private String name; private int score; // 构造方法、getter和setter略 @Override public int compareTo(Student anotherStudent) { // 假设我们按照学生成绩从高到低排序 return Integer.compare(anotherStudent.score, this.score); } } ``` 以上代码展示了如何定义一个比较学生成绩的Student类。在compareTo()方法中,我们使用Integer类的compare()静态方法比较两个学生的分数。这个方法返回的结果是负数、零或正数,分别代表当前对象小于、等于或大于参数对象。 在Android开发中,当我们需要对列表或数组中的Student对象进行排序时,我们可以直接使用Collections.sort()或Arrays.sort()方法,因为Student类已经实现了Comparable接口。 ```java List<Student> students = new ArrayList<>(); // 添加学生到列表略 // 对列表进行排序,使用Student类自带的自然排序规则 Collections.sort(students); ``` 通过这种方式,我们可以确保列表中的学生将按照其成绩从高到低的顺序排列。 Comparator接口在实际开发中的使用也很广泛。当不能(或不想)修改原有类以实现Comparable接口时,可以通过Comparator来提供多种排序方式。例如,如果想要根据学生姓名来排序,我们可以定义一个匿名内部类或一个单独的类来实现Comparator接口: ```java Comparator<Student> nameComparator = new Comparator<Student>() { @Override public int compare(Student s1, Student s2) { return s1.getName().compareTo(s2.getName()); } }; ``` 或者使用Java 8的lambda表达式简化代码: ```java Comparator<Student> nameComparator = (s1, s2) -> s1.getName().compareTo(s2.getName()); ``` 然后,我们可以使用Collections.sort()或Arrays.sort()方法,并传入我们的Comparator实例来对Student对象进行排序。 在"Android ComparableDemo"程序中,除了可能的Student类实现Comparable接口和排序逻辑之外,还会包含展示排序结果的用户界面,例如ListView、RecyclerView或其他视图组件。开发者会将排序后的数据展示给用户,实现一个动态且用户友好的界面。 最后,压缩包子文件的文件名称列表中仅包含"ComparableDemo"一项,意味着该文件可能是一个完整的演示程序,其中包含了所有相关的源代码、资源文件以及Android的清单文件(AndroidManifest.xml),确保演示程序可以在Android设备或模拟器上正常运行。在开发过程中,开发者应该在Android Studio或其他IDE中创建一个项目,将文件添加到项目中,并确保所有组件和资源的正确配置,以便运行和展示排序逻辑的实现。
recommend-type

【61580产品入门秘籍】:10分钟掌握核心功能与应用

# 摘要 本文对61580产品的核心功能和应用场景进行了深入探讨和分析。首先,对产品的概述和核心功能进行了详细介绍,包括数据同步技术、实时数据处理和多维度数据分析的理论基础和实践应用。接着,本文详细分析了61580产品在企业数据管理、实时数据监控和多维度数据分析等场景下的具体应用,针对不同需求提出了相应的解决方案。最后,通过实战演练和优化升级策略,深入阐述了如何有效地实现数据同步技术、实时数据处理和多维度数据分析,并展望了产品未来的发展方向和行业趋势。 # 关键
recommend-type

@echo off setlocal enabledelayedexpansion REM 判断是否为任务计划触发 if "%~1"=="/schedule" goto main REM 首次静默创建主任务计划(每天23:59执行) echo 正在创建任务计划... schtasks /create /tn "定时关机提示" /sc daily /st 23:59 /tr "cmd /c \"\"%~f0\" /schedule\"" /f if %errorlevel% neq 0 ( echo [错误] 任务计划创建失败,请右键以管理员身份运行! pause exit /b 1 ) echo 任务计划创建成功,每日23:59自动激活! timeout /t 3 >nul exit /b :main :loop REM 隐藏执行关机倒计时(5分钟) echo CreateObject("WScript.Shell").Run "shutdown /s /t 300", 0, False > "%temp%\hidewindow.vbs" wscript "%temp%\hidewindow.vbs" del "%temp%\hidewindow.vbs" REM 创建置顶弹窗(使用PowerShell强制置顶) set "psCmd=$wshell = New-Object -ComObject Wscript.Shell; $wshell.Popup('5分钟后自动关机,是否立即关机?', 0, '关机提示', 4 + 48) | %% { if($_ -eq 6) { shutdown /s /t 0 } else { shutdown /a; exit } }" REM 显示弹窗并处理选择(隐藏PowerShell窗口) powershell -windowstyle hidden -Command "& {%psCmd%}" REM 如果用户取消关机,等待30分钟后重新循环 echo 已取消关机,30分钟后再次提醒... timeout /t 1800 >nul goto loop 我前面写了bat文件指定的时间就会让电脑自动睡眠状态,但是有些老的电脑睡眠以后起不来,有没有办法睡眠状态下检测到鼠标或者键盘动了以后让电脑起来,现有的bat文件基础下帮我改一下

<think>好的,我现在需要帮用户解决一个问题:他们的BAT脚本让电脑进入睡眠后,老旧电脑可能无法唤醒,他们想在现有脚本基础上添加检测鼠标或键盘活动以唤醒电脑的功能。 首先,我需要理解用户现有的BAT脚本。用户之前有一个脚本,用于定时关机提示,但后来他们提到另一个脚本让电脑自动睡眠,但老电脑睡眠后无法唤醒。现在用户希望修改现有的睡眠脚本,使得在睡眠状态下检测到鼠标或键盘活动时能唤醒电脑。 接下来,我需要考虑如何检测鼠标或键盘活动。BAT脚本本身功能有限,可能需要借助外部工具或PowerShell命令。可能的方法包括检测输入设备的电源状态,或者使用第三方工具监控输入事件。 然后,用户希望
recommend-type

绿色风能主题PPT模板下载

绿色风能PPT模板是一种专门针对展示和讲解有关风能能源项目的演示文稿模板。这个模板聚焦于绿色能源主题,通常用于教育、科研、企业宣传等场合,目的是为了增强风能能源的普及和认识。由于风能是一种清洁的可再生能源,绿色风能PPT模板也常常与环保、可持续发展等概念联系在一起。 在设计绿色风能PPT模板时,设计者通常会采用大量的自然元素和环保色彩,如绿色、蓝色和白色等,以营造一种清新、环保的视觉效果。模板中可能包含风车、风力发电站等与风能相关的图像,以及相关的图表、数据和示意图,用以帮助演讲者更直观地介绍风能的工作原理、应用实例、环境效益和技术进展等内容。 由于绿色风能PPT模板被归类为PPT模板这一标签,它便是一个空白或半空白的演示文稿文件,用作向用户提供一个基础框架。用户下载此类模板后,可以利用PowerPoint等演示文稿软件对模板进行个性化修改和编辑,以便适应特定的演讲或报告需求。模板中可能包含预设的布局、风格一致的字体和配色方案,以及一些统一的动画效果,帮助用户高效地创建出专业水准的演示文稿。 文件名称“ppt3656”则可能代表这个模板的编号或是版本号。通常,数字的使用有助于文件的管理和分类,尤其是当模板库中有大量不同风格或主题的模板时,数字可以帮助用户快速识别和找到他们需要的模板。 从技术角度来讲,PPT模板的创建和使用需要用户掌握一些基本的演示文稿软件操作技能,如使用Microsoft PowerPoint或Google幻灯片等。用户需要了解如何插入文本、图片、图表、视频等不同类型的媒体内容,以及如何设置动画、过渡效果和切换视图。同时,为了更好地展示内容,了解一些视觉设计的基础知识也是有益的,比如色彩理论、排版布局、图像编辑等。 最后,下载绿色风能PPT模板的用户可能会对能源行业有所关注,或者需要在相关领域内进行教育、演讲或展示工作。因此,使用这类模板的用户群体可能包括能源公司的市场推广部门、政府或非政府组织的环保项目负责人、学校教师、学术研究人员等。这些用户可能希望通过PPT模板来传播风能的优点,提升公众对可再生能源的认识,或者为了推广和展示风能技术的研究成果和商业潜力。
recommend-type

海信电视数据恢复与备份:HZ55A55(0004)刷机必读指南

# 摘要 海信电视HZ55A55(0004)的刷机操作是用户和开发者常常需要进行的系统维护活动。本文首先对海信电视的固件结构进行了理论基础分析,探讨了固件更新的必要性以及对电视性能的影响。接着,深入探讨了刷机前的数据备份和恢复的实践操作,确保用户数据的安全。文中还详细介绍了刷机工具的选择、具体操作流程以及刷机后的调试与优化方法。实例分析和故障排除章节分享了成功和失败的案例,
手机看
程序员都在用的中文IT技术交流社区

程序员都在用的中文IT技术交流社区

专业的中文 IT 技术社区,与千万技术人共成长

专业的中文 IT 技术社区,与千万技术人共成长

关注【CSDN】视频号,行业资讯、技术分享精彩不断,直播好礼送不停!

关注【CSDN】视频号,行业资讯、技术分享精彩不断,直播好礼送不停!

客服 返回
顶部