Android Device Monitor

背景

最近在看《第一行代码》(第二版)中关于数据存储方案的介绍。数据的状态分为两种:瞬时状态和持久状态,一般保存在内存中的数据随着活动的关闭,数据也就销毁了,如果我们想保存这些数据,该怎么办呢?书中介绍了三种实现数据持久化的方法,分别是:文件存储,SharedPreference 存储以及数据库存储。

在学习第一种数据持久化方法文件存储的时候,我们通过 Context 类中的 openFileOutput() 方法创建一个指定了保存路径的 FileOutputStream 对象 out,然后用这个 out 对象去创建一个 OutputStreamWriter 对象,之后再用这个 OutputStreamWriter 对象创建一个 BufferedWriter 对象 writer,我们就是通过这个 writer 的 write() 方法向文件输出流写入我们想要保存的数据的。文件写入数据之后默认保存在 /data/data/(package name)/files 目录下。我们可以通过 Android Studio 中的 Android Device Monitor 去查看,Android Device Monitor 的打开方法如下所示。

打开 Android Device Monitor

问题1:设备 Offline, File Explorer 空空如也

打开 Android Device Monitor 之后,发现里面什么都没有,左边的面板上显示了一台模拟器,但处于 offline 状态,不对呀,我的模拟器明明开着的呀,它正在运行我刚刚写的 app 呢,怎么这里会显示离线状态呢?于是我试着 Google 去找解决方案。

File Explorer 不显示文件

在 StackOverflow 上我发现了有人和我一样也碰到了这个问题,不过底下的回答寥寥无几,我找到了一个最高赞的答案(其实也就一个赞….),如下图所示,他的意思是他也不明白其中的具体原理,但是他发现如果先打开 Android Device Monitor,然后再运行模拟器,就可以解决这个问题。

解决 File Explorer 不显示文件(from SO)

于是我按照这个方法尝试,先打开 Android Device Monitor,里面什么也没有,然后打开模拟器,随着模拟器的启动,Monitor 左侧面板出现了 offline 的模拟器,然后再相继启动很多服务,最后就显示为 online 了,如下图所示,这样我们就可以通过 Monitor 中的 File Explorer 标签访问模拟器中的文件了。

解决 File Explorer 不显示文件

问题2:File Explorer 中的文件无法访问

还记得刚刚说的吗,openFileOutput() 方法生成的文件存放在一个默认路径,即 /data/data/(package name)/files 下。由于我的包名是 com.knightaoko.filepersistencetest,所以在下文的叙述中,大家看到这个陌生的字符串不要惊讶,实际操作下替换成你自己的包名即可。(这期间还碰到个问题,好像 Android Device Monitor 不能和 Instant Run 同时运行,所以在此需要把 Instant Run 功能关闭,怎么关闭大家自行 Google,很简单的。)于是,下一步当然是打开这个路径了,找到 data,点击,没反应,再点击,还是没反应。又遇到问题了,我无法访问 data 目录,通过 Google 搜索,我知道了 File Explorer 中的每个文件和文件夹都是有访问权限的,如下图红框中标识的就是 data 文件夹的访问权限。

无权访问

访问权限是 drwxrwx–x,这玩意是什么意思呢?我相信,学过 Linux 的同学肯定一眼就知道这是什么意思。不知道也没关系,听我慢慢道来,这一个字符串一共有 10 个字符,第一个字符表示是文件夹还是文件,如果第 1 个字符是 d,则表示是文件夹。后面 9 个字符可以分为三组,第一组 rwx 表示所有者(user)对文件的访问权限,r 表示可读(read),w 表示可写(write),x 表示可被执行,- 表示没有该权限;第二组 rwx 表示组群(group)对文件的访问权限;第三组 –x 表示其他人(other)对文件的访问权限,可以看到,没有 r 和 w,说明 data 对其他人来说是不可读不可写的,怪不得我们不能打开它呢,那怎么办呢?于是我又网上搜啊搜,发现可以通过一定的方法修改文件的访问权限的,对,这个方法就是 adb shell。

adb shell 是 Android SDK 中的一个工具,你可以在你的 Android SDK 目录下的 platform-tools 找到它,下图就是我电脑上的该目录:

adb 所在目录

我们需要将这个目录添加到环境变量,为的是后面在命令行直接调用 adb 中的命令。比如在此我将 D:\AndroidSDK\platform-tools 添加到环境变量(具体怎么添加应该不用手把手教学了吧)后,运行 cmd 命令行,然后输入下面的命令。

通过 adb shell 获取文件访问权限

PS:(如果有多个设备,会出现错误 “more than one device/emulator”,可采用如下方式:

1、adb devices -l    :(该命令会查出当前所有设备并列出)

2、adb -s model:you_model shell  :(该命令是指定某个设备进行adb脚本)

*** PS over 然后接着改权限 ***

)

  • adb shell 是打开 adb 外壳程序。
  • su 是获取 root 权限,只有 root 权限才能修改文件的访问权限。
  • 底下一堆 chmod 777 是把对应的文件或文件夹的访问权限的后 9 位设置成 rwxrwxrwx,777 就三个二进制 111, 合起来就是 111111111(9 个 1),即相当于把后 9 位访问权限都置 1,于是就成了 rwxrwxrwx,即所有用户都具有该文件的全部访问权限(妈妈再也不用担心我打不开 data 了…)。我们来看看结果:

修改后的文件访问权限

问题3:File Explorer 中的文件无法 pull(下载)

可以看到这些目录都能访问了,可以在包名目录下看到生成的 data 文件了,下一步我们想把这个文件复制到我们的电脑里,怎么操作呢?选中该文件,点击右上角的 pull 图标即可,如下图所示:

文件下载按钮

点击之后,选好保存路径。结果我们发现文件并没有保存下来,在 Console 窗口还提示了错误:

文件下载出错

意思就是拉取文件失败,因为你无权进行该操作。又是一脸懵逼,怎么办?还能怎么办,继续 Google 呗!这里我找了很久,终于发现了解决方案:adb root 。没错,在命令行输入这条语句,完美解决!(注意是在 windows 下的命令行,而不是 adb shell 里,当然前提是你添加了 adb 的目录到你的环境变量里)

通过 adb root 获取文件下载权限

然后我们再来 pull 一下,卧槽,崩溃了,又发现问题了:Failed to pull selection。

文件下载再次出错

怎么办?OK,继续 Google,在 StackOverflow 里找到了一个高赞的解决方法:

重新选择模拟器(from SO)

什么意思?重新选择一下 Android Device Monitor 左侧面板的设备即可,没错!就是把下图红框中国的玩意儿点一下。

重新选择模拟器

好了,我们再找到刚刚那个要拷贝的文件,pull 一下,卧槽!!成功了,真是踏破铁鞋无觅处,得来全得靠姑姑(谷歌)。

文件下载成功

总结

总结一下吧,初次使用 Android Device Monitor,本来只是用来辅助学习 SQLite,结果一上午碰到一大堆问题,总算都一一解决了,在此总结一下。

  1. Android Device Monitor 要在模拟器运行之前就打开,这样才不会出现模拟器设备 offline 的问题。

  2. Android Device Monitor 和 Instant Run 貌似有冲突,在用 Android Device Monitor 时关闭 Instant Run 功能。我相信还有更好的解决方法,不然打开一个大一点的 APP,不能同时用这两个功能,那多影响效率啊。只是我目前还处于学习阶段,上述方案能解决我当前的问题,不影响我进一步学习,那它就是 OK 的。

  3. 操作 Android Device Monitor 时,应该默认把你当做其他人(other)对待,所以你对文件的访问权限很受限制。解决方法是在命令行中,通过在 adb shell 里获取 root 权限,并更改相应文件的访问权限,当然记得将 adb 所在目录添加进环境变量。

  4. 能访问了但是不能下载也是很麻烦的事,通过在命令行运行 adb root 获取文件的下载权限

  5. 通过上述操作之后,就能正常使用 Android Device Monitor 进行文件管理了。但不幸的是,当你重启模拟器,或启动另一台模拟器时,这些配置又得重新来过,否则你还是无法访问和下载模拟器中的文件。这个问题的根源在于权限二字,问题不是不能获取权限(通过 adb shell 可以获取 root 权限),而在于获取权限太麻烦了,希望后续版本的 Android Studio 可以直接在 Android Device Monitor 的 GUI 中设置相应获取权限的命令,使得广大 Android 开发者对模拟器中的文件操作更有效率。

  6. Google 大法好

  7. 最后还有一点,我目前是跟着《第一行代码》(第二版)学习 Android 开发的,但是作者郭霖老师没有组织读者交流群,所以在学习过程中遇到问题时,要么不知所措,要么大费周折才解决一个问题。如果有志同道合的同学一起交流,那样学习更有效率,遇到问题也能大家一起讨论解决。不知道大家有没有《第二行代码》学习社群推荐,或者有意向和我一起组建一个这样的学习环境,大家可以在留言区畅所欲言

  8. PS:本人初出茅庐,开发经验匮乏。如果你发现上文中有不对的地方,还望不吝赐教!

  9. (转载者PS:) 再打开设备监控时有时打不开,说端口占用,也困扰了我一下,解决办法:

  10. dos指令:netstat -ano,列出所有端口的情况。在列表中我们观察被占用的端口,比如是49157,首先找到它。
  11. 查看被占用端口对应的PID,输入命令:netstat -aon|findstr "49157",回车,记下最后一位数字,即PID,这里是2720

  12. 继续输入tasklist|findstr "2720",回车,查看是哪个进程或者程序占用了2720端口,结果是:svchost.exe

  13. 我们打开任务管理器,切换到进程选项卡,在PID一列查看2720对应的进程是谁,然后干掉它,问题解决。