Android Q 的 Scoped Storage 和 “沙盒”(早就没了)所有你需要知道的

2019-09-04 16:09:19 +08:00
 RikkaW

Scoped Storage ( Beta 3 起至今)

官方文档 中的 "An app that uses scoped storage always has read/write access to the files that it creates, both inside and outside its app-specific directory." 这句话相当容易误解。其中的 "outside its app-specific directory" 并不是指应用可以像以前有存储权限时一样可以任意读写内置存储,而只是应用可以通过 Media Store 和 Storage Access Framework 在其专有文件夹以外建立文件并可任意访问。

简而言之,使用 Scoped Storage 的行为如下:

使用 Media Store 插入图片的例子

private boolean insertImage(File image) throws IOException {
    ContentValues values;
    // 向 Media Store 插入标记为待定的空白文件
    values = new ContentValues();
    values.put(MediaStore.Images.ImageColumns.RELATIVE_PATH, Environment.DIRECTORY_PICTURES + "test"); // 不同类型文件可用 RELATIVE_PATH 不用,具体请参阅 MediaProvider 源码
    values.put(MediaStore.Images.ImageColumns.DISPLAY_NAME, Long.toString(System.currentTimeMillis()));
    values.put(MediaStore.Images.ImageColumns.IS_PENDING, true);
    Uri uri = getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
    if (uri == null) {
        return false;
    }
    // 写入文件内容
    InputStream is = new FileInputStream(image);
    OutputStream os = getContentResolver().openOutputStream(uri, "rw");
    byte[] b = new byte[8192];
    for (int r; (r = is.read(b)) != -1; ) {
        os.write(b, 0, r);
    }
    os.flush();
    os.close();
    is.close();
    // 移除待定标记,其他应用可访问该文件
    values = new ContentValues();
    values.put(MediaStore.Images.ImageColumns.IS_PENDING, false);
    return getContentResolver().update(uri, values, null, null) == 1;
}

“沙盒”(只存在于 Beta 2,Beta 5 起删除相关代码

Google 的“沙盒”的做法和 Rikka 的 Storage Redirect 核心部分原理相同,即使用挂载将一个只属于应用的文件夹(Android/sandbox)挂载为内置存储根目录,这样应用就只能访问该文件夹而不能访问真实的内置存储。

但这样做显然会产生一些问题(包括但不限于):

Google 在 Beta 2 的“沙盒”使用和 Rikka 一样的思路,即在最小化影响的前提下实现隔离应用产生的文件的功能。但结果是大部分用户根本不 care 有没有隔离(毕竟喜欢乱吐文件的应用几乎都来自中国大陆),只会关心自己要使用的应用有没有炸裂。

个人觉得对 Google 来说,于其使用这样相对激进的方案到头来吃力不讨好,倒不如选择之后的更简单的 Scoped Storage,并给应用一个大版本的时间进行适配,因为毕竟哪种方案都需要应用开发者做出相同的适配工作。<del>然而现在文档既模糊又不全(</del>

个人感慨

应用滥用存储权限的问题自古以来就是一个大问题,Rikka 个人从小时候(划掉)起就一直想解决这个问题。所以 Rikka 从 2016 年底开始创造 Storage Redirect,至今已经有相当高的完成度,并且对各种问题都有相应的解决方案,只要使用者愿意稍微动一动脑子就可以获得相当好的体验。

2019 年,Google 终于愿意开始解决这个问题,根据 Rikka 个人的实验和体验,绝大部分应用需要做的工作并没有很多,只是目前 Google 的文档过于模糊和残缺。

只要 Google 在 Android R 时能真正推行所有应用必须作出改变,那 R 到来的一两年世界后将会变得更美好(划掉)。(当然如果有应用滥用 Storage Access Framework 的授权访问整个存储 Rikka 魔法还是有用的(划掉

对某些过于乐观的人的疑惑(

从 Beta 2 起,就不断地听到有人说“啊,有了 Q 就没有应用乱吐文件的问题啦”,Beta 2 之后砍掉沙盒也有人哀嚎“啊,怎么砍了”。然而即使保留,除非所有应用都进行适配,否则必然是无尽的“找不到文件”“打不开文件”。国内大厂的适配时间都是以年计算(想想 QQ WeChat 今年才支持接收 content uri ),所以即使予以保留,也肯定是要等待至少一两年后才能有比较好的体验(

13480 次点击
所在节点    Android
31 条回复
Love4Taylor
2019-09-04 16:11:28 +08:00
rikka 大法好!
z919126592
2019-09-04 16:16:23 +08:00
是 rikka 诶
photon006
2019-09-04 16:17:47 +08:00
今天升级 android 10 就遇到很多存储相关问题,play store 无法下载 app 更新,无法截图,微信无法选择图片发送给好友,photos 无法查看照片,很多网友也遇到:

https://support.google.com/photos/thread/13511562?hl=en

https://support.google.com/photos/thread/13518838?hl=en
huaxianyan
2019-09-04 16:25:09 +08:00
Rikka 的钱钱又飞回来了

今天用上了 Android 10,官方可以允许切换到三键模式好评,就是应用不能写剪贴板让我有点难受,其他设备给手机传点文本还要推送个通知过去再复制
7654
2019-09-04 16:27:31 +08:00
不给权限不给用
gz911122
2019-09-04 16:28:33 +08:00
是 rikka 耶
Buges
2019-09-04 16:29:28 +08:00
有一点不敢苟同的是,如果“沙盒”保留的话“找不到文件”等问题仅在最开始会存在,在国产厂商争相追 Android 版本的情况下,除非国内定制 ROM 魔改砍掉,否则定然会倒逼 app 做适配(大厂适配慢那是无足轻重的小问题,这种严重影响体验的 bug 不可能不跟进),不然新手机微信之类的都不能用了,怎么可能?
虽然存储重定向的魔法确实好用,但从根本上让应用不再这么干不是更好么?
起码到了 Q 终于不再需要 xprivacy 来阻止应用获取 IMEI 了,hook 一时爽,应用不再申请了当然更爽。
expy
2019-09-04 16:36:02 +08:00
乐观的想法:不适配就崩溃,会倒逼应用适配啊。
RikkaW
2019-09-04 16:44:36 +08:00
@Buges 然而确实是可以选择完全不开启这个功能的(
deorth
2019-09-04 16:48:44 +08:00
是大佬,awsl
tankren
2019-09-04 16:55:21 +08:00
膜拜大佬 正在使用 Storage Redirect
有个问题啊,微信启用了重定向之后 微博国际版的图片不能直接分享到微信了 分享操作可以无误的完成但是图片并没有发出去
HankAviator
2019-09-04 16:59:09 +08:00
逼一波大厂也得妥协,微信就是拖到不匹配通知渠道就从 play 下架的死线前不久才更新适配,之前拿各种借口搪塞,必须改时不也痛快。
s82kd92l
2019-09-04 18:09:07 +08:00
好像 beta6 文档里面写了有新的 appops 类型可以强制 scoped storage, 不过正式发布的文档里面找不到这个描述了。不知道代码里面还有没有
s82kd92l
2019-09-04 18:14:51 +08:00
RikkaW
2019-09-04 18:33:20 +08:00
@s82kd92l 相关的东西早就看过了 开了也没意义 “我求你了读一读第一段 Scoped Storage 的行为”
(一种你就是最后讲的那种人的感觉
momocraft
2019-09-04 18:36:34 +08:00
希望对国内厂家有效,play 版的 QQ 现在都不上架了...
little_cup
2019-09-04 18:37:29 +08:00
@Buges 除了 Google,其他设备商只能逼小厂,大厂不愿适配那就只能乖乖 ROM 里写死给他们开白名单。
echo314
2019-09-04 19:17:30 +08:00
@little_cup Google 在这点考虑上可能并不是顾忌大厂不愿适配,而是因为直接强上影响用户体验,不管是厂商还是 Google 都会被用户骂。
国内大厂不愿适配这点影响不大,Google 对海外安卓市场有极强的控制力,除非国内大厂想放弃海外市场,联合国内厂商搞白名单才有意义,毕竟面向海外的还是适配,既然都得适配为什么要搞两套呢?
efsg
2019-09-04 19:27:34 +08:00
不明觉厉 简单来说就是存储权限被废掉,不管怎么样都不能读写外部存储吗
RikkaW
2019-09-04 19:29:10 +08:00
@efsg 先读官方文档再读这个.. 现在是 target 29 才有(并且可以 opt-out ),计划是下个大版本强制

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/597899

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX