android用java和c實現(xiàn)查找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++ 編寫的共享庫回調(diào)取得全部 SD 卡路徑的函數(shù)
public String
GetAllSDPath() throws TokenException
{
String strMountInfo = "";
// 1.首先獲得系統(tǒng)已加載的文件系統(tǒng)信息
try
{
// 創(chuàng)建系統(tǒng)進程生成器對象
ProcessBuilder objProcessBuilder = new ProcessBuilder();
// 執(zhí)行 mount -h 可以看到 mount : list mounted filesystems
// 這條命令可以列出已加載的文件系統(tǒng)
objProcessBuilder.command( "mount" ); // 新的操作系統(tǒng)程序和它的參數(shù)
// 設(shè)置錯誤輸出都將與標(biāo)準(zhǔn)輸出合并
objProcessBuilder.redirectErrorStream( true );
// 基于當(dāng)前系統(tǒng)進程生成器的狀態(tài)開始一個新進程,并返回進程實例
Process objProcess = objProcessBuilder.start();
// 阻塞線程至到本地操作系統(tǒng)程序執(zhí)行結(jié)束,返回本地操作系統(tǒng)程序的返回值
objProcess.waitFor();
// 得到進程對象的輸入流,它對于進程對象來說是已與本地操作系統(tǒng)程序的標(biāo)準(zhǔn)輸出流(stdout)相連接的
InputStream objInputStream = objProcess.getInputStream();
byte[] buffer = new byte[1024];
// 讀取 mount 命令程序返回的信息文本
while ( -1 != objInputStream.read( buffer ) )
{
strMountInfo = strMountInfo + new String( buffer );
}
// 關(guān)閉進程對象的輸入流
objInputStream.close();
// 終止進程并釋放與其相關(guān)的任何流
objProcess.destroy();
}
catch ( Exception e )
{
e.printStackTrace();
}
// 2.然后再在系統(tǒng)已加載的文件系統(tǒng)信息里查找 SD 卡路徑
// mount 返回的已加載的文件系統(tǒng)信息是以一行一個信息的形式體現(xiàn)的,
// 所以先用換行符拆分字符串
String[] lines = strMountInfo.split( "\n" );
// 清空該字符串對象,下面將用它來裝載真正有用的 SD 卡路徑列表
strMountInfo = "";
for ( int i = 0;
i < lines.length;
i++ )
{
// 如果該行內(nèi)有 /mnt/和 vfat 字符串,說明可能是內(nèi)/外置 SD 卡的掛載路徑
if ( -1 != lines[i].indexOf( " /mnt/" ) && // 前面要有空格,以防斷章取義
-1 != lines[i].indexOf( " vfat " ) ) // 前后均有空格
{
// 再以空格分隔符拆分字符串
String[] blocks = lines[i].split( "\\s" ); // \\s 為空格字符
for ( int j = 0;
j < blocks.length;
j++ )
{
// 如果字符串中含有/mnt/字符串,說明可能是我們要找的 SD 卡掛載路徑
if ( -1 != blocks[j].indexOf( "/mnt/" ) )
{
// 排除重復(fù)的路徑
if ( -1 == strMountInfo.indexOf( blocks[j] ) )
{
// 用分號符(;)分隔 SD 卡路徑列表,
strMountInfo += blocks[j] + ";";
}
}
}
}
}
return strMountInfo;
}
C 版代碼如下:
char caStdOutLine[1024]; // mount 命令的標(biāo)準(zhǔn)輸出中的一行信息
char* pcTmpSDPath = NULL;
// 再用 mount 命令獲得的找身份認(rèn)證鎖
do // 非循環(huán),只是為了方便控制分支層次,便于控制分支流向
{
// 通過創(chuàng)建一個管道,調(diào)用 fork 產(chǎn)生一個子進程,
// 執(zhí)行一個 shell 以運行命令來開啟一個進程。
// 這個進程必須由 pclose() 函數(shù)關(guān)閉。
FILE* fp = popen( "mount", // 一個指向以 NULL 結(jié)束的 shell 命令字符串的指針,
// 這行命令將被傳到 bin/sh 并使用 -c 標(biāo)志,
// 那么 shell 將執(zhí)行這個命令從這個字符串中讀取。
"r" ); // 文件指針連接到 shell 命令的標(biāo)準(zhǔn)輸出
if ( NULL == fp )
{
break;
}
while( NULL != fgets( caStdOutLine,
sizeof( caStdOutLine ),
fp ) )
{
// 如果 找到了你想要的 SD 卡掛載路徑 的話,則
if ( 判斷條件 )
{
// 注:管道中的數(shù)據(jù)一定要讀完,不然會崩潰掉的
continue; // 就不再試下一個掛載地址了
}
// 如果該行內(nèi)有 /mnt/和 vfat 字符串,說明可能是內(nèi)/外置 SD 卡的掛載路徑
if ( NULL == strstr( caStdOutLine, " /mnt/" ) && // 前面要有空格,以防斷章取義
NULL == strstr( caStdOutLine, " /storage/" ) ) // 前面要有空格,以防斷章取義
{
continue; // 不滿足條件說明這行不是內(nèi)/外置 SD 卡的掛載路徑
}
if ( NULL == strstr( caStdOutLine, " vfat " ) ) // 前后均有空格
{
continue; // 不滿足條件說明這行不是內(nèi)/外置 SD 卡的掛載路徑
}
// 再以空格分隔符拆分字符串
pcTmpSDPath = strtok( caStdOutLine, " " );
do // 這里是循環(huán),嘗試每一個路徑
{
if ( ( NULL == pcTmpSDPath ) ||
( '\0' == *pcTmpSDPath ) )
{
continue;
}
// 如果字符串中含有/mnt/字符串,說明可能是我們要找的 SD 卡掛載路徑
if ( NULL == strstr( pcTmpSDPath, "/mnt/" ) &&
NULL == strstr( pcTmpSDPath, "/storage/" ) )
{
continue;
}
// TODO: 在此添加對 SD 卡路徑使用的語句,如果只是用其中一個,別忘了設(shè)置已找到想要 SD 卡路徑的標(biāo)識
}while ( pcTmpSDPath = strtok( NULL, " " ) );
}
// 關(guān)閉標(biāo)準(zhǔn) I/O 流,等待命令執(zhí)行結(jié)束,然后返回 shell 的終止?fàn)顟B(tài)。
// 如果 shell 不能被執(zhí)行,
// 則 pclose() 返回的終止?fàn)顟B(tài)與 shell 已執(zhí)行 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 <label> <mount_point> <part> <sysfs_path1...>
## label - Label for the volume
## mount_point - Where the volume will be mounted
## part - Partition # (1 based), or 'auto' for first usable partition.
## <sysfs_path> - List of sysfs paths to source devices
######################
# external sd card
dev_mount sdcard-ext /mnt/sdcard-ext auto /devices/platform/sdhci-tegra.2/mmc_host/mmc1 /devices/platform/sdhci-tegra.2/mmc_host/mmc2
# flash drive connection to USB1
dev_mount usbdisk_1.0 /mnt/usbdisk_1.0 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1:1.0
# flash drive connection through hub connected to USB1
dev_mount usbdisk_1.1 /mnt/usbdisk_1.1 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1.1
dev_mount usbdisk_1.2 /mnt/usbdisk_1.2 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1.2
dev_mount usbdisk_1.3 /mnt/usbdisk_1.3 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1.3
dev_mount usbdisk_1.4 /mnt/usbdisk_1.4 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1.4
dev_mount usbdisk_1.5 /mnt/usbdisk_1.5 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1.5
dev_mount usbdisk_1.6 /mnt/usbdisk_1.6 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1.6
dev_mount usbdisk_1.7 /mnt/usbdisk_1.7 auto /devices/platform/tegra-ehci.0/usb2/2-1/2-1.7
C 版代碼如下:
char caStdOutLine[1024]; // cat 命令的標(biāo)準(zhǔn)輸出中的一行信息
char* pcTmpSDPath = NULL;
char* pcNotSpace = NULL;
// 用 /system/etc/vold.fstab 獲得的 SD 卡路徑找身份認(rèn)證鎖
do // 非循環(huán),只是為了方便控制分支層次,便于控制分支流向
{
// 通過創(chuàng)建一個管道,調(diào)用 fork 產(chǎn)生一個子進程,
// 執(zhí)行一個 shell 以運行命令來開啟一個進程。
// 這個進程必須由 pclose() 函數(shù)關(guān)閉。
FILE* fp = popen( "cat /system/etc/vold.fstab", // 一個指向以 NULL 結(jié)束的 shell 命令字符串的指針,
// 這行命令將被傳到 bin/sh 并使用 -c 標(biāo)志,
// 那么 shell 將執(zhí)行這個命令從這個字符串中讀取。
"r" ); // 文件指針連接到 shell 命令的標(biāo)準(zhǔn)輸出
if ( NULL == fp )
{
break;
}
while( NULL != fgets( caStdOutLine,
sizeof( caStdOutLine ),
fp ) )
{
// 如果 找到了你想要的 SD 卡掛載路徑 的話,則
if ( 判斷條件 )
{
// 注:管道中的數(shù)據(jù)一定要讀完,不然會崩潰掉的
continue; // 就不再試下一個掛載地址了
}
// Format: dev_mount <label> <mount_point> <part> <sysfs_path1...>
// 去除開頭的空格
pcNotSpace = caStdOutLine + strspn( caStdOutLine, " " );
if ( NULL == pcNotSpace ||
'\0' == *pcNotSpace ||
'#' == *pcNotSpace || // 行首字符為#說明是注釋行
'd' != pcNotSpace[0] || // 句子開頭不是 dev_mount
'e' != pcNotSpace[1] ||
'v' != pcNotSpace[2] ||
'_' != pcNotSpace[3] ||
'm' != pcNotSpace[4] ||
'o' != pcNotSpace[5] ||
'u' != pcNotSpace[6] ||
'n' != pcNotSpace[7] ||
't' != pcNotSpace[8] )
{
continue; // 不滿足條件說明這行不是內(nèi)/外置 SD 卡的掛載路徑
}
// 再以空格分隔符拆分字符串
pcTmpSDPath = strtok( pcNotSpace, " " );
do // 這里是循環(huán),嘗試每一個路徑
{
if ( ( NULL == pcTmpSDPath ) ||
( '\0' == *pcTmpSDPath ) )
{
continue;
}
// 如果字符串中含有/mnt/字符串,說明可能是我們要找的 SD 卡掛載路徑
if ( NULL == strstr( pcTmpSDPath, "/mnt/" ) &&
NULL == strstr( pcTmpSDPath, "/storage/" ) )
{
continue;
}
// TODO: 在此添加對 SD 卡路徑使用的語句,如果只是用其中一個,別忘了設(shè)置已找到想要 SD 卡路徑的標(biāo)識
}while ( pcTmpSDPath = strtok( NULL, " " ) );
}
// 關(guān)閉標(biāo)準(zhǔn) I/O 流,等待命令執(zhí)行結(jié)束,然后返回 shell 的終止?fàn)顟B(tài)。
// 如果 shell 不能被執(zhí)行,
// 則 pclose() 返回的終止?fàn)顟B(tài)與 shell 已執(zhí)行 exit 一樣。
pclose( fp );
}while ( 0 );
- Android編程獲取SD卡路徑及剩余容量的方法
- Android獲取SD卡路徑及SDCard內(nèi)存的方法
- android 手機SD卡讀寫操作(以txt文本為例)實現(xiàn)步驟
- Android獲取SD卡中選中圖片的路徑(URL)示例
- Android調(diào)用相機并將照片存儲到sd卡上實現(xiàn)方法
- Android獲取assets文件夾中的數(shù)據(jù)并寫入SD卡示例
- android將Bitmap對象保存到SD卡中的方法
- Eclipse開發(fā)環(huán)境導(dǎo)入android sdk的sample中的源碼
- Android 獲取內(nèi)外SD卡路徑幾種方法總結(jié)
相關(guān)文章
Android Handler主線程和一般線程通信的應(yīng)用分析
本篇文章小編為大家介紹,Android Handler主線程和一般線程通信的應(yīng)用分析。需要的朋友參考下2013-04-04Android開發(fā)中使用achartengine繪制各種圖表的方法
這篇文章主要介紹了Android開發(fā)中使用achartengine繪制各種圖表的方法,結(jié)合具體實例形式分析了Android基于圖表生成類庫achartengine進行圖表繪制的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-10-10使用RecyclerView添加Header和Footer的方法
RecyclerView雖然作為ListView的替代者有著較好的性能提升,但是ListView的一些常用功能卻沒有提供,比如我們平時會經(jīng)常用到的addHeaderView,addFooterView,既然RecyclerView沒有提供這個方法,我們應(yīng)該如何為列表添加頭部和底部呢,接下來通過本文給大家介紹2016-03-03Android開發(fā)中實現(xiàn)發(fā)送短信的小程序示例
這篇文章主要介紹了Android開發(fā)中實現(xiàn)發(fā)送短信的小程序示例,文中還附帶了一個監(jiān)聽廣播接收者的升級版短信發(fā)送例子,需要的朋友可以參考下2016-04-04