自我介绍:

网易 Android 专家工程师,网易云音乐 Android 负责人,主导从零开发了网易云音乐 Android

客户端,目前是杭州研究院专业委员会成员,负责每年的评级,规范起草,面试招聘等相关工作。参与并制作了网易云课堂 Android

微专业相关课程,反响不错。

Android的存储卡分为两类,一类叫内置存储(Internal Storage),还有一类叫外置存储(External Storage)。内置存储除了系统文件以外,就是我们程序的私有工作目录,context.getFilesDir就是指向该目录,私有目录默认情况下面只有自己能读写。外置存储是个公共目录,所有程序都可以读写(这里指主外存储,还有个叫副外存储后面会讲),对应Envirnment.getExternalStorageDirectory。

存储卡设计有两个阶段:

Android4.4版本之前有内置存储,可以插外置存储卡

内置存储分配出一定区域来模拟外置存储卡,同时还可以再插外置存储卡。这种情况下面外存储有主副之分。前者叫Primary External Storage,后者叫Secondary External Storage。

国内厂家对存储系统的修改造成的兼容问题主要有以下几个点:

在Android4.4之前还不支持副外存储的情况下面,厂家进行了扩展,也可以多插一张存储卡来实现手机扩容。这种情况下面没有标准API来获取这张存储卡。

第一点里面多出来的副外存储权限设计上面跟4.4系统以后支持的副外存储不一样。厂家扩展出来的副外存储和主外存储一样任何程序可读可写。而4.4系统以后支持的副外存储任何程序有可读权限,但无法写入,默认这个目录context.getExternalFilesDirs(Environment.DIRECTORY_xxx)可以写入,但需要注意这个目录会随程序卸载而被自动删除。不管哪个版本主外存储都是任意程序可读可写的。

某些厂家的系统里面(比如华为),主副外置存储可以进行互换

4.4以上系统主副外存储卡的权限设计为什么有区别也是一直不能理解的事情,可能主外存储为了保持高低系统版本的兼容就一直维持原样,而高版本上面为了收敛存储卡上面乱七八糟的写入内容就对新增的副外存储做了限制(包括不能随便写入一个目录,卸载时尽量清理运行产生的垃圾)。整个系统越来越向IOS靠拢。关键主外存储还是维持原样是已经放弃的意思吗?

The WRITE_EXTERNAL_STORAGE permission must only grant write access to the primary external storage on a device. Apps must not be allowed

to write to secondary external storage devices, except in their

package-specific directories as allowed by synthesized permissions.

Restricting writes in this way ensures the system can clean up files

when applications are uninstalled.

下面我们来详细说明下遇到的问题和解决办法

首先是获取存储卡,对于Android4.4之前扩展出来的副外存储,因为没有标准API来获取,所以必须另辟蹊径。常见的比较靠谱的有两种方法

1.使用StorageManager反射调用getVolumeList方法,Environment.getExternalStorageDirectory的实现就是用这个hide方法,所以还是比较靠谱,通过这个方法拿到的设备还可以判断是否已挂载(mounted),如图所示三星的手机上面有主副存储卡,第三个设备能显示但是未挂载状态,代码上面可以进行忽略。对于厂家自己扩展副外存储卡的情况不排除这个StorageManager修改上面存在问题:

读写内存卡权限申请失败 Android csdn_android 内存卡读写

2.使用mount命令输出进行解析,mount输出如图:

读写内存卡权限申请失败 Android csdn_Storage_02

mount命令本身很靠谱,难点在于如何判断哪些设备是真正的用户外置存储卡,这里提供代码片段供参考:

public static List getExternalMounts() {
final List out = new ArrayList();
if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
out.add(Environment.getExternalStorageDirectory().getPath());
}
String reg = ".* (vfat|ntfs|exfat|fat32|fuse|fuseblk|texfat|sdcardfs|esdfs|(fuse\\.\\S+)) .*rw.*";
BufferedReader br = null;
try {
final Process process = new ProcessBuilder().command("mount").redirectErrorStream(true).start();
process.waitFor();
br = new BufferedReader(new InputStreamReader(process.getInputStream()));
String line;
while ((line = br.readLine()) != null) {
if (!line.toLowerCase(Locale.US).contains("asec") && !line.toLowerCase(Locale.US).contains("obb") && !line.toLowerCase(Locale.US).contains("secure")) {
if (line.matches(reg)) {
String[] parts = line.split(" ");
for (int i = 1; i < parts.length; i++) {
String part = parts[i];
if (part.startsWith("/")) {
if (new File(part).canRead() && !checkIfSameSdcard(part, out)) {
out.add(part);
}
break;
}
}
}
}
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
File[] externalFilesDirs = NeteaseMusicApplication.getInstance().getExternalFilesDirs(Environment.DIRECTORY_DOCUMENTS);
if (externalFilesDirs != null) {
for (int i = 1; i < externalFilesDirs.length; i++) {
File file = externalFilesDirs[i];
if (file == null) {
continue;
}
String sdPath = file.getAbsolutePath().substring(0, file.getAbsolutePath().indexOf("/Android/"));
boolean alreadyIn = checkIfSameSdcard(sdPath, out);
if (!alreadyIn) {
out.add(sdPath);
}
}
}
}
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
if (br != null) {
br.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
return out;
}

首先直接把主外存储卡加进来, 然后对mount命令输出每行进行正则匹配,再进行可读判断检查再加进来。为了增加兼容,对于4.4以后系统还同时使用系统API来获取避免遗漏。这个只是示例代码,实际比如华为的系统互换了主副外存储要特别增加测试。

和虾米酷狗的同学交流,据他们反馈使用方法一兼容性还不错也实现简单。出于历史原因我们网易云音乐目前还是在使用方法二(从我们使用经验来看还算稳定),如果有mount命令解析上面的问题可以和我交流。 获取外置存储卡的这条路就是不断的收到反馈不断修改兼容的过程。

而对于4.4系统以后支持的副外存储卡普通目录无法写入问题,我们代码可以做检测然后使用context.getExternalFilesDirs(Environment.DIRECTORY_xxx)目录。这种做法在4.4以前厂家自己扩展出来的副外存储也是兼容的(本来就全部可读可写)。如果确实有对副外存储卡进行写入需求,可以使用DocumentsUI(这是一个framework里面的内置程序,使用新的Storage Access Framework框架实现)来实现授权后进行写入。这里还有一篇华为关于4.4系统副外存储卡的兼容问题,可以看看。

对于能够互换主副外存储卡的系统,只能多加测试来验证是否兼容了。如果有必要的话需要增加检查另外一张存储卡上面有没有自己的工作目录和数据来进行处理。