热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

android用java和c实现查找sd卡挂载路径(sd卡路径)的方法

这篇文章主要介绍了android用java和c实现查找sd卡挂载路径(sd卡路径)的方法,需要的朋友可以参考下

方法一:

分析 mount 命令的返回信息,例如:

代码如下:

$ mount
rootfs / rootfs ro,relatime 0 0
tmpfs /dev tmpfs rw,nosuid,relatime,mode=755 0 0
devpts /dev/pts devpts rw,relatime,mode=600 0 0
proc /proc proc rw,relatime 0 0
sysfs /sys sysfs rw,relatime 0 0
debugfs /sys/kernel/debug debugfs rw,relatime 0 0
none /acct cgroup rw,relatime,cpuacct 0 0
tmpfs /mnt/asec tmpfs rw,relatime,mode=755,gid=1000 0 0
tmpfs /mnt/obb tmpfs rw,relatime,mode=755,gid=1000 0 0
none /dev/cpuctl cgroup rw,relatime,cpu 0 0
/dev/block/platform/sdhci-tegra.3/by-name/system /system ext4 ro,relatime,barrier=1,data=ordered 0 0
/dev/block/platform/sdhci-tegra.3/by-name/userdata /data ext4 rw,nosuid,nodev,noatime,barrier=1,data=ordered 0 0
/dev/block/platform/sdhci-tegra.3/by-name/cache /cache ext4 rw,nosuid,nodev,noatime,barrier=1,data=ordered 0 0
/dev/block/platform/sdhci-tegra.3/by-name/pdsb /pds ext2 ro,relatime 0 0
/dev/fuse /mnt/sdcard fuse rw,nosuid,nodev,relatime,user_id=1023,group_id=1023,default_permissions,allow_other 0 0
/dev/block/vold/179:9 /mnt/sdcard-ext vfat rw,dirsync,nosuid,nodev,noexec,relatime,uid=1000,gid=1015,fmask=0702,dmask=0702,allow_utime=0020,codepage=cp437,iocharset=iso8859-1,shortname=mixed,utf8,errors=remount-ro 0 0
/dev/block/vold/179:9 /mnt/secure/asec vfat rw,dirsync,nosuid,nodev,noexec,relatime,uid=1000,gid=1015,fmask=0702,dmask=0702,allow_utime=0020,codepage=cp437,iocharset=iso8859-1,shortname=mixed,utf8,errors=remount-ro 0 0
tmpfs /mnt/sdcard-ext/.android_secure tmpfs ro,relatime,size=0k,mode=000 0 0

Java 版代码如下:
// 给 C/C++ 编写的共享库回调取得全部 SD 卡路径的函数
public String
GetAllSDPath() throws TokenException
{
    String strMountInfo = "";

    // 1.首先获得系统已加载的文件系统信息
    try
    {
        // 创建系统进程生成器对象
        ProcessBuilder objProcessBuilder = new ProcessBuilder();
        // 执行 mount -h 可以看到 mount : list mounted filesystems
        // 这条命令可以列出已加载的文件系统
        objProcessBuilder.command( "mount" ); // 新的操作系统程序和它的参数
        // 设置错误输出都将与标准输出合并
        objProcessBuilder.redirectErrorStream( true );
        // 基于当前系统进程生成器的状态开始一个新进程,并返回进程实例
        Process objProcess = objProcessBuilder.start();
        // 阻塞线程至到本地操作系统程序执行结束,返回本地操作系统程序的返回值
        objProcess.waitFor();

        // 得到进程对象的输入流,它对于进程对象来说是已与本地操作系统程序的标准输出流(stdout)相连接的
        InputStream objInputStream = objProcess.getInputStream();

        byte[] buffer = new byte[1024];

        // 读取 mount 命令程序返回的信息文本
        while ( -1 != objInputStream.read( buffer ) )
        {
            strMountInfo = strMountInfo + new String( buffer );
        }

        // 关闭进程对象的输入流
        objInputStream.close();

        // 终止进程并释放与其相关的任何流
        objProcess.destroy();
    }
    catch ( Exception e )
    {
        e.printStackTrace();
    }

    // 2.然后再在系统已加载的文件系统信息里查找 SD 卡路径  
    // mount 返回的已加载的文件系统信息是以一行一个信息的形式体现的,
    // 所以先用换行符拆分字符串
    String[] lines = strMountInfo.split( "\n" );

    // 清空该字符串对象,下面将用它来装载真正有用的 SD 卡路径列表
    strMountInfo = "";

    for ( int i = 0;
              i               i++ )
    {
        // 如果该行内有 /mnt/和 vfat 字符串,说明可能是内/外置 SD 卡的挂载路径
        if ( -1 != lines[i].indexOf( " /mnt/" ) && // 前面要有空格,以防断章取义
             -1 != lines[i].indexOf( " vfat " ) )  // 前后均有空格
        {
            // 再以空格分隔符拆分字符串
            String[] blocks = lines[i].split( "\\s" ); // \\s 为空格字符
            for ( int j = 0;
                      j                       j++ )
            {
                // 如果字符串中含有/mnt/字符串,说明可能是我们要找的 SD 卡挂载路径
                if ( -1 != blocks[j].indexOf( "/mnt/" ) )
                {
                    // 排除重复的路径
                    if ( -1 == strMountInfo.indexOf( blocks[j] ) )
                    {
                        // 用分号符(;)分隔 SD 卡路径列表,
                        strMountInfo += blocks[j] + ";";
                    }
                }
            }
        }
    }

    return strMountInfo;
}

C 版代码如下:

代码如下:

char caStdOutLine[1024]; // mount 命令的标准输出中的一行信息
char* pcTmpSDPath = NULL;

// 再用 mount 命令获得的找身份认证锁
do // 非循环,只是为了方便控制分支层次,便于控制分支流向
{
    // 通过创建一个管道,调用 fork 产生一个子进程,
    // 执行一个 shell 以运行命令来开启一个进程。
    // 这个进程必须由 pclose() 函数关闭。
    FILE* fp = popen( "mount", // 一个指向以 NULL 结束的 shell 命令字符串的指针,
                               // 这行命令将被传到 bin/sh 并使用 -c 标志,
                               // 那么 shell 将执行这个命令从这个字符串中读取。
                      "r" );   // 文件指针连接到 shell 命令的标准输出

    if ( NULL == fp )
    {
        break;
    }

    while( NULL != fgets( caStdOutLine,
                          sizeof( caStdOutLine ),
                          fp ) )
    {
        // 如果 找到了你想要的 SD 卡挂载路径 的话,则
        if ( 判断条件 )
        {
            // 注:管道中的数据一定要读完,不然会崩溃掉的
            continue; // 就不再试下一个挂载地址了
        }

        // 如果该行内有 /mnt/和 vfat 字符串,说明可能是内/外置 SD 卡的挂载路径
        if ( NULL == strstr( caStdOutLine, " /mnt/" ) &&    // 前面要有空格,以防断章取义
             NULL == strstr( caStdOutLine, " /storage/" ) ) // 前面要有空格,以防断章取义
        {
            continue; // 不满足条件说明这行不是内/外置 SD 卡的挂载路径
        }

        if ( NULL == strstr( caStdOutLine, " vfat " ) )  // 前后均有空格
        {
            continue; // 不满足条件说明这行不是内/外置 SD 卡的挂载路径
        }

        // 再以空格分隔符拆分字符串
        pcTmpSDPath = strtok( caStdOutLine, " " );

        do // 这里是循环,尝试每一个路径
        {
            if ( ( NULL == pcTmpSDPath ) ||
                 ( '\0' == *pcTmpSDPath ) )
            {
                continue;
            }

            // 如果字符串中含有/mnt/字符串,说明可能是我们要找的 SD 卡挂载路径
            if ( NULL == strstr( pcTmpSDPath, "/mnt/" ) &&
                 NULL == strstr( pcTmpSDPath, "/storage/" ) )
            {
                continue;
            }

            // TODO: 在此添加对 SD 卡路径使用的语句,如果只是用其中一个,别忘了设置已找到想要 SD 卡路径的标识

        }while ( pcTmpSDPath = strtok( NULL, " " ) );
    }

    // 关闭标准 I/O 流,等待命令执行结束,然后返回 shell 的终止状态。
    // 如果 shell 不能被执行,
    // 则 pclose() 返回的终止状态与 shell 已执行 exit 一样。
    pclose( fp );

}while ( 0 );



方法二:
分析 cat /system/etc/vold.fstab 命令的返回信息,例如:
代码如下:

$ cat /system/etc/vold.fstab
## Vold 2.0 fstab for Stingray
#######################
## Regular device mount
##
## Format: dev_mount


推荐阅读
  • 解决Anaconda安装TensorFlow时遇到的TensorBoard版本问题
    本文介绍了在使用Anaconda安装TensorFlow时遇到的“Could not find a version that satisfies the requirement tensorboard”错误,并提供详细的解决方案,包括创建虚拟环境和配置PyCharm项目。 ... [详细]
  • 如何将本地Docker镜像推送到阿里云容器镜像服务
    本文详细介绍将本地Docker镜像上传至阿里云容器镜像服务的步骤,包括登录、查看镜像列表、推送镜像以及确认上传结果。通过本文,您将掌握如何高效地管理Docker镜像并将其存储在阿里云的镜像仓库中。 ... [详细]
  • 在项目部署后,Node.js 进程可能会遇到不可预见的错误并崩溃。为了及时通知开发人员进行问题排查,我们可以利用 nodemailer 插件来发送邮件提醒。本文将详细介绍如何配置和使用 nodemailer 实现这一功能。 ... [详细]
  • 本文将探讨Java编程语言中对象和类的核心概念,帮助读者更好地理解和应用面向对象编程的思想。通过实际例子和代码演示,我们将揭示如何在Java中定义、创建和使用对象。 ... [详细]
  • 本文详细探讨了JavaScript中的作用域链和闭包机制,解释了它们的工作原理及其在实际编程中的应用。通过具体的代码示例,帮助读者更好地理解和掌握这些概念。 ... [详细]
  • 本文详细介绍了如何在 Android 开发中高效地管理和使用资源,包括本地资源和系统资源的访问方法。通过实例和代码片段,帮助开发者更好地理解和应用资源管理的最佳实践。 ... [详细]
  • Python 内存管理机制详解
    本文深入探讨了Python的内存管理机制,涵盖了垃圾回收、引用计数和内存池机制。通过具体示例和专业解释,帮助读者理解Python如何高效地管理和释放内存资源。 ... [详细]
  • C#设计模式学习笔记:观察者模式解析
    本文将探讨观察者模式的基本概念、应用场景及其在C#中的实现方法。通过借鉴《Head First Design Patterns》和维基百科等资源,详细介绍该模式的工作原理,并提供具体代码示例。 ... [详细]
  • 本文详细介绍了划分树这一数据结构,重点探讨了其在子树和中值计算中的应用及优化方法。 ... [详细]
  • Android Studio 安装与配置指南
    本教程详细介绍了如何下载并安装 Android Studio,包括设置 SDK 路径和优化启动性能的方法。通过这些步骤,您可以顺利地开始开发 Android 应用。 ... [详细]
  • Appium + Java 自动化测试中处理页面空白区域点击问题
    在进行移动应用自动化测试时,有时会遇到某些页面没有返回按钮,只能通过点击空白区域返回的情况。本文将探讨如何在Appium + Java环境中有效解决此类问题,并提供详细的解决方案。 ... [详细]
  • 如何清除Chrome浏览器地址栏的特定历史记录
    在使用Chrome浏览器时,你可能会发现地址栏保存了大量浏览记录。有时你可能希望删除某些特定的历史记录而不影响其他数据。本文将详细介绍如何单独删除地址栏中的特定记录以及批量清除所有历史记录的方法。 ... [详细]
  • 利用Selenium与ChromeDriver实现豆瓣网页全屏截图
    本文介绍了一种使用Selenium和ChromeDriver结合Python代码,轻松实现对豆瓣网站进行完整页面截图的方法。该方法不仅简单易行,而且解决了新版Selenium不再支持PhantomJS的问题。 ... [详细]
  • 嵌入式开发环境搭建与文件传输指南
    本文详细介绍了如何为嵌入式应用开发搭建必要的软硬件环境,并提供了通过串口和网线两种方式将文件传输到开发板的具体步骤。适合Linux开发初学者参考。 ... [详细]
  • 解决TensorFlow CPU版本安装中的依赖问题
    本文记录了在安装CPU版本的TensorFlow过程中遇到的依赖问题及解决方案,特别是numpy版本不匹配和动态链接库(DLL)错误。通过详细的步骤说明和专业建议,帮助读者顺利安装并使用TensorFlow。 ... [详细]
author-avatar
海尔兄弟
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有