资讯专栏INFORMATION COLUMN

Android我还可以相信你多少系列文章五之存储卡

caige / 766人阅读

我即将在2017.7.8号开一个直播讲堂,感兴趣的同学点击快来参加吧:https://segmentfault.com/l/15...
内容包括:

Android 知识体系分享

从入门到提高的学习路径

如何进一步突破瓶颈,进一步提升

充足的时间和大家讨论,回答大家问题

自我介绍:

网易 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修改上面存在问题:

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


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系统副外存储卡的兼容问题,可以看看。

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

更多文章请关注微信公众号:anzhuozhimei

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/13690.html

相关文章

  • Android我还可以相信多少系列文章一之推送

    摘要:于是演变出了很多变种,每个厂家自研一套,比如小米,华为都是对外公开。根据我们的经验同时接入小米和华为,在华为手机上面走华为推送,在其他手机上面走小米推送。还要警惕的是,接入推送之后有可能存在消息延迟,这完全依赖于推送服务提供商的后台服务。 我即将在2017.7.8号开一个直播讲堂,感兴趣的同学点击快来参加吧:https://segmentfault.com/l/15...内容包括: ...

    seanlook 评论0 收藏0
  • Android我还可以相信多少系列文章四之悬浮窗

    摘要:事实上从某一次跟官方的现场交流会上,官方人员也承认悬浮窗事实上也不是设计给应用用的。在上面特别增加了来满足悬浮窗需求。在使用的时候,系统还会进行提示是否允许该应用显示悬浮窗。我们可以显示之后再来进行兼容。 我即将在2017.7.8号开一个直播讲堂,感兴趣的同学点击快来参加吧:https://segmentfault.com/l/15...内容包括: Android 知识体系分享 从...

    JowayYoung 评论0 收藏0
  • Android我还可以相信多少系列文章三之通知栏

    摘要:加上的通知还能提升进程优先级,大有被滥用的趋势。自己保活不了不算,还要别人拉起来帮忙通知后台起另外一个程序的用法也被比如华为系统限制。通知栏不显示高版本的原生系统自带通知栏显示管理功能,默认是允许显示。接下来我想聊一聊通知栏的跳转设计。 我即将在2017.7.8号开一个直播讲堂,感兴趣的同学点击快来参加吧:https://segmentfault.com/l/15...内容包括: ...

    bingchen 评论0 收藏0
  • Android我还可以相信多少系列文章二之音视频播放

    摘要:我们也可以使用结合或者来实现视频播放,本质和是一样的,不过有更多的灵活性。主流的音视频播放器大部分都是在这个上面进行改造的。 我即将在2017.7.8号开一个直播讲堂,感兴趣的同学点击快来参加吧:https://segmentfault.com/l/15...内容包括: Android 知识体系分享 从入门到提高的学习路径 如何进一步突破瓶颈,进一步提升 充足的时间和大家讨论,回答...

    tommego 评论0 收藏0
  • Javascript数组系列五之增删改和强大的 splice()

    摘要:删除数组元素的开始索引需要删除元素的个数,插入数组的元素语法因为参数变化多样,我们主要从三个方面来展示的用法。 今天是我们介绍数组系列文章的第五篇,也是我们数组系列的最后一篇文章,只是数据系列的结束,所以大家不用担心,我们会持续的更新干货文章。 生命不息,更新不止! 今天我们就不那么多废话了,直接干货开始。 我们在《Javascript数组系列一之栈与队列》中描述我们是如何利用 pus...

    chavesgu 评论0 收藏0

发表评论

0条评论

caige

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<