Android 获取SD卡路径:

外置sd卡路径,也许很多同学在平时的工作中并不会用到,因为现在很多机型都不支持外置sd卡(这也是Google目标),所以并不用考虑外置sd卡的路径问题。除了开发文件管理类的应用之外,其他应用使用 Enviroment 这个类中的一些静态方法就能满足需要。但也有一些特殊需求需要用到外置sd卡路径,那怎么才能准确获得外置sd卡的路径呢?
方法一
//内置sd卡路径
String sdcardPath = System.getenv("EXTERNAL_STORAGE");
//内置sd卡路径
String sdcardPath = Environment.getExternalStorageDirectory().getAbsolutePath();
//外置置sd卡路径
String extSdcardPath = System.getenv("SECONDARY_STORAGE");
在Enviroment类的源码中获得sd卡路径其实也是通过 System.getnv() 方法来实现的,如隐藏的方法:
/** {@hide} */
public static File getLegacyExternalStorageDirectory() {
return new File(System.getenv(ENV_EXTERNAL_STORAGE));
}
注:更详细的内容还是去看Enviroment源码。
另外要注意的是,在API 23版本中 SECONDARY_STORAGE 被移除。
方法二
private static String getStoragePath(Context mContext, boolean is_removale) {
StorageManager mStorageManager = (StorageManager) mContext.getSystemService(Context.STORAGE_SERVICE);
Class<?> storageVolumeClazz = null;
try {
storageVolumeClazz = Class.forName("android.os.storage.StorageVolume");
Method getVolumeList = mStorageManager.getClass().getMethod("getVolumeList");
Method getPath = storageVolumeClazz.getMethod("getPath");
Method isRemovable = storageVolumeClazz.getMethod("isRemovable");
Object result = getVolumeList.invoke(mStorageManager);
final int length = Array.getLength(result);
for (int i = 0; i < length; i++) {
Object storageVolumeElement = Array.get(result, i);
String path = (String) getPath.invoke(storageVolumeElement);
boolean removable = (Boolean) isRemovable.invoke(storageVolumeElement);
if (is_removale == removable) {
return path;
}
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
return null;
}
通过反射的方式使用在sdk中被 隐藏 的类 StroageVolume 中的方法getVolumeList(),获取所有的存储空间(Stroage Volume),然后通过参数is_removable控制,来获取内部存储和外部存储(内外sd卡)的路径,参数 is_removable为false时得到的是内置sd卡路径,为true则为外置sd卡路径。
在API 23 Enviroment 类中的内部类 UserEnvironment 中有一方法getExternalDirs与此一样,代码如下:
public File[] getExternalDirs() {
final StorageVolume[] volumes = StorageManager.getVolumeList(mUserId,StorageManager.FLAG_FOR_WRITE);
final File[] files = new File[volumes.length];
for (int i = 0; i < volumes.length; i++) {
files[i] = volumes[i].getPathFile();
}
return files;
}
再看Enviroment的getExternalStorageDirectory方法实现:
public static File getExternalStorageDirectory() {
throwIfUserRequired();
return sCurrentUser.getExternalDirs()[0];
}
可以看出,在API 23时,先是通过getExternalDirs()获取到所有存储空间的File[]数组,这个数组的第一个值:getExternalDirs()[0],即为内置sd卡所在路径。
而在API 23 之前的版本中,并没有类似getExternalDirs()的方法通过StorageVolume直接获得存储空间(Storage Volume),而时通过别的方式来实现的,看关键方法的源码:
public static File getExternalStorageDirectory() {
throwIfUserRequired();
return sCurrentUser.getExternalDirsForApp()[0];
}
这里的 getExternalDirsForApp() 和上面的 getExternalDirs() 的作用是一样的,都是得到所有存储空间的File[]数组。
public File[] getExternalDirsForApp() {
return mExternalDirsForApp;
}
mExternalDirsForApp 是在 Enviroment 类中的内部类 UserEnvironment 的构造方法中初始化的,Enviroment#UserEnvironment构造函数源码如下:
public UserEnvironment(int userId) {
// See storage config details at http://source.android.com/tech/storage/
String rawExternalStorage = System.getenv(ENV_EXTERNAL_STORAGE);
String rawEmulatedSource = System.getenv(ENV_EMULATED_STORAGE_SOURCE);
String rawEmulatedTarget = System.getenv(ENV_EMULATED_STORAGE_TARGET);
String rawMediaStorage = System.getenv(ENV_MEDIA_STORAGE);
if (TextUtils.isEmpty(rawMediaStorage)) {
rawMediaStorage = "/data/media";
}
ArrayList<File> externalForVold = Lists.newArrayList();
ArrayList<File> externalForApp = Lists.newArrayList();
if (!TextUtils.isEmpty(rawEmulatedTarget)) {
// Device has emulated storage; external storage paths should have
// userId burned into them.
final String rawUserId = Integer.toString(userId);
final File emulatedSourceBase = new File(rawEmulatedSource);
final File emulatedTargetBase = new File(rawEmulatedTarget);
final File mediaBase = new File(rawMediaStorage);
// /storage/emulated/0
externalForVold.add(buildPath(emulatedSourceBase, rawUserId));
externalForApp.add(buildPath(emulatedTargetBase, rawUserId));
// /data/media/0
mEmulatedDirForDirect = buildPath(mediaBase, rawUserId);
} else {
// Device has physical external storage; use plain paths.
if (TextUtils.isEmpty(rawExternalStorage)) {
Log.w(TAG, "EXTERNAL_STORAGE undefined; falling back to default");
rawExternalStorage = "/storage/sdcard0";
}
// /storage/sdcard0
externalForVold.add(new File(rawExternalStorage));
externalForApp.add(new File(rawExternalStorage));
// /data/media
mEmulatedDirForDirect = new File(rawMediaStorage);
}
// Splice in any secondary storage paths, but only for owner
final String rawSecondaryStorage = System.getenv(ENV_SECONDARY_STORAGE);
if (!TextUtils.isEmpty(rawSecondaryStorage) && userId == UserHandle.USER_OWNER) {
for (String secondaryPath : rawSecondaryStorage.split(":")) {
externalForVold.add(new File(secondaryPath));
externalForApp.add(new File(secondaryPath));
}
}
mExternalDirsForVold = externalForVold.toArray(new File[externalForVold.size()]);
mExternalDirsForApp = externalForApp.toArray(new File[externalForApp.size()]);
}
也可以根据这个方法得到一个获取所有存储空间的路径的方法getStorageDirectories():
/**
* Returns all available SD-Cards in the system (include emulated)
* <p/>
* Warning: Hack! Based on Android source code of version 4.3 (API 18)
* Because there is no standard way to get it.
* TODO: Test on future Android versions 4.4+
*
* @return paths to all available SD-Cards in the system (include emulated)
*/
private static final Pattern DIR_SEPARATOR = Pattern.compile("/");
public List<String> getStorageDirectories() {
// Final set of paths
final ArrayList<String> rv = new ArrayList<String>();
// Primary physical SD-CARD (not emulated)
final String rawExternalStorage = System.getenv("EXTERNAL_STORAGE");
// All Secondary SD-CARDs (all exclude primary) separated by ":"
final String rawSecondaryStoragesStr = System.getenv("SECONDARY_STORAGE");
// Primary emulated SD-CARD
final String rawEmulatedStorageTarget = System.getenv("EMULATED_STORAGE_TARGET");
if (TextUtils.isEmpty(rawEmulatedStorageTarget)) {
// Device has physical external storage; use plain paths.
if (TextUtils.isEmpty(rawExternalStorage)) {
// EXTERNAL_STORAGE undefined; falling back to default.
rv.add("/storage/sdcard0");
} else {
rv.add(rawExternalStorage);
}
} else {
// Device has emulated storage; external storage paths should have
// userId burned into them.
final String rawUserId;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN_MR1) {
rawUserId = "";
} else {
final String path = Environment.getExternalStorageDirectory().getAbsolutePath();
final String[] folders = DIR_SEPARATOR.split(path);
final String lastFolder = folders[folders.length - 1];
boolean isDigit = false;
try {
Integer.valueOf(lastFolder);
isDigit = true;
} catch (NumberFormatException ignored) {
}
rawUserId = isDigit ? lastFolder : "";
}
// /storage/emulated/0[1,2,...]
if (TextUtils.isEmpty(rawUserId)) {
rv.add(rawEmulatedStorageTarget);
} else {
rv.add(rawEmulatedStorageTarget + File.separator + rawUserId);
}
}
// Add all secondary storages
if (!TextUtils.isEmpty(rawSecondaryStoragesStr)) {
// All Secondary SD-CARDs splited into array
final String[] rawSecondaryStorages = rawSecondaryStoragesStr.split(File.pathSeparator);
Collections.addAll(rv, rawSecondaryStorages);
}
rootmode = Sp.getBoolean("rootmode", false);
if (rootmode)
rv.add("/");
File usb = getUsbDrive();
if (usb != null && !rv.contains(usb.getPath())) rv.add(usb.getPath());
return rv;
}
public File getUsbDrive() {
File parent;
parent = new File("/storage");
try {
for (File f : parent.listFiles()) {
if (f.exists() && f.getName().toLowerCase().contains("usb") && f.canExecute()) {
return f;
}
}
} catch (Exception e) {
}
parent = new File("/mnt/sdcard/usbStorage");
if (parent.exists() && parent.canExecute())
return (parent);
parent = new File("/mnt/sdcard/usb_storage");
if (parent.exists() && parent.canExecute())
return parent;
return null;
}
综上分析,通过方法一和方法二都可以正确的获取内外sd卡路径,但方法一会存在以下问题:
1、API>=23 时方法一无效(暂未测试)
2、有些厂商的Rom改动太多,对相关原生API的支持存在问题,这时方法一可能会存在问题。
3、其他一些情况造成的原因(基本与2差不多,是ROM等因素造成的)
所以,在使用时建议使用方法二来获取内外置sd卡路径,在API 23(Android 6.0)之前使用getStorageDirectories() 应该也是OK的。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
# Android
# 获得内外SD卡路径几种方法
# 内外SD卡路径获得
# SD卡路径
# Android获取SD卡路径及SDCard内存的方法
# android读取sdcard路径下的文件的方法
# Android 遍历SDCARD的文件夹并显示目录信息
# Android获取内置sdcard跟外置sdcard路径
# Android6.0 storage目录sd卡存储的路径创建详解
# Android获取SDcard目录及创建文件夹的方法
# 外置
# 的是
# 类中
# 时方
# 来实现
# 存在问题
# 都是
# 是在
# 太多
# 都不
# 第一个
# 就能
# 中有
# 而在
# 要注意
# 去看
# 希望能
# 但也
# 与此
# 可以看出
相关文章:
如何通过wdcp面板快速创建网站?
建站之星如何配置系统实现高效建站?
专业公司网站制作公司,用什么语言做企业网站比较好?
道歉网站制作流程,世纪佳缘致歉小吴事件,相亲网站身份信息伪造该如何稽查?
较简单的网站制作软件有哪些,手机版网页制作用什么软件?
建站之星展会模板:智能建站与自助搭建高效解决方案
c++ stringstream用法详解_c++字符串与数字转换利器
建站之星×万网:智能建站系统+自助建站平台一键生成
建站主机选购指南:核心配置优化与品牌推荐方案
深入理解Android中的xmlns:tools属性
如何快速搭建支持数据库操作的智能建站平台?
高性价比服务器租赁——企业级配置与24小时运维服务
昆明高端网站制作公司,昆明公租房申请网上登录入口?
深圳 网站制作,深圳招聘网站哪个比较好一点啊?
建站为何优先选择香港服务器?
Swift中switch语句区间和元组模式匹配
如何选择靠谱的建站公司加盟品牌?
建站上传速度慢?如何优化加速网站加载效率?
如何设计高效校园网站?
如何快速启动建站代理加盟业务?
高端建站三要素:定制模板、企业官网与响应式设计优化
制作网站的公司有哪些,做一个公司网站要多少钱?
建站主机与虚拟主机有何区别?如何选择最优方案?
寿县云建站:智能SEO优化与多行业模板快速上线指南
网站app免费制作软件,能免费看各大网站视频的手机app?
公司网站的制作公司,企业网站制作基本流程有哪些?
手机网站制作与建设方案,手机网站如何建设?
如何彻底卸载建站之星软件?
沈阳个人网站制作公司,哪个网站能考到沈阳事业编招聘的信息?
网站制作知乎推荐,想做自己的网站用什么工具比较好?
西安大型网站制作公司,西安招聘网站最好的是哪个?
c# await 一个已经完成的Task会发生什么
招商网站制作流程,网站招商广告语?
建站10G流量真的够用吗?如何应对访问高峰?
在线流程图制作网站手机版,谁能推荐几个好的CG原画资源网站么?
网站制作企业,网站的banner和导航栏是指什么?
开封网站制作公司,网络用语开封是什么意思?
金*站制作公司有哪些,金华教育集团官网?
建站之星IIS配置教程:代码生成技巧与站点搭建指南
如何用AWS免费套餐快速搭建高效网站?
C++用Dijkstra(迪杰斯特拉)算法求最短路径
建站之星会员如何解锁更多建站功能?
公司网站制作价格怎么算,公司办个官网需要多少钱?
如何通过VPS搭建网站快速盈利?
c# 服务器GC和工作站GC的区别和设置
韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南
大连网站制作费用,大连新青年网站,五年四班里的视频怎样下载啊?
免费ppt制作网站,有没有值得推荐的免费PPT网站?
美食网站链接制作教程视频,哪个教做美食的网站比较专业点?
简单实现Android验证码
*请认真填写需求信息,我们会在24小时内与您取得联系。