很多安装模拟器都带了adb.exe,有的是在根目录,有的四在tool下,有的改了名字,比如夜神安装模拟器就把adb.exe改成了nox_adb.exe,而逍遥安卓和雷电安卓模拟器都保留android-sdk中的adb.exe,但我们比较的时候,这些adb.exe明显大小不一样,也就是说在开发安卓模拟器时二次封装了adb.exe.
错误描述
当我们运行安卓模拟器时,除了启动自身进程外,都会启动adb.exe,比如当我们启动雷电模拟器时,系统进程里会出现adb.exe
和dnplayer.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-server
和adb start-server
处理了这个问题,那么这样独开一个雷电模拟器的时候,返回的是
127.0.0.1:5555 device
其实这也是成功的,这就是第一个模拟器,如果我们先接受了雷电模拟器自启动的adb.exe进程,再次启动adb.exe时,这时候返回的就是
emulator-5554 device
那么我们如何处理这个问题,到底那个最好,在模块初始化中,可以都加上,这样万无一失。