很多安装模拟器都带了adb.exe,有的是在根目录,有的四在tool下,有的改了名字,比如夜神安装模拟器就把adb.exe改成了nox_adb.exe,而逍遥安卓和雷电安卓模拟器都保留android-sdk中的adb.exe,但我们比较的时候,这些adb.exe明显大小不一样,也就是说在开发安卓模拟器时二次封装了adb.exe.

错误描述

当我们运行安卓模拟器时,除了启动自身进程外,都会启动adb.exe,比如当我们启动雷电模拟器时,系统进程里会出现adb.exednplayer.exe两个进程。

这时候我再运行

> adb devices


就会返回 List of devices attached emulator-5554 device 127.0.0.1:5555 offline


明明开了一个模拟器,为何返回一个连接不上的127.0.0.1:5555 ?当我们测试逍遥安卓时就不会出现,只出现在雷电模拟器上。

错误分析

1、 adb 启动就连接5555端口
启动 adb 的时候, adb 通过 "adb fork-server server" 启动 adb demon
而后demon 就会去找本地的 5555 端口, 直到 5555+32
2、为何连接叫 emulator-5554 而不是 emulator-5555
为何连接上就叫 emulator, 这是因为 adb 期望自动为用户连接本机的emulator ( 可以多达16个)
这是因为缺省emulator的 console 端口是 5554 ( 应该可以用 telnet 连接与 emulator 交互(还没有试验)) , 而adb 的端口是console端口 +1 就是 5555
所以当有程序监听 5555 端口, 会被 adb 认为是 emulator

解决办法:

总思路很简单,先杀死adb.exe进程,再次启动adb.exe.
结束adb.exe进程的办法

一)通过cmd命令结束占用端口进程。

1、5037为adb默认端口
2、查看占用端口的进程PID


> netstat -aon|findstr 5037
  TCP    127.0.0.1:5037         0.0.0.0:0              LISTENING       15448


3、通过PID查看所有进程


> tasklist /fi "PID eq 15448"
 
映像名称                       PID 会话名              会话#       内存使用
========================= ======== ================ =========== ============
adb.exe                      15448 Console                    1      5,728 K


4、杀死占用端口的进程

> taskkill /pid 15448 /f


成功: 已终止 PID 为 15448 的进程。


二)使用api(TerminateProcess)结束进程(adb.exe)

缺点:5037的端口未必是adb.exe占用。比如酷狗,360,电脑管家,很多软件都集成了手机助手,也就是说,启动这些软件的时候,可能就启动了adb,exe,进程名称未必是adb,exe,

三)最佳解决方案

其实,我们在adb模块第三课,adb初始化里,已经通过 adb kill-serveradb start-server处理了这个问题,那么这样独开一个雷电模拟器的时候,返回的是

127.0.0.1:5555 device

其实这也是成功的,这就是第一个模拟器,如果我们先接受了雷电模拟器自启动的adb.exe进程,再次启动adb.exe时,这时候返回的就是

emulator-5554   device

那么我们如何处理这个问题,到底那个最好,在模块初始化中,可以都加上,这样万无一失。