Chrome浏览器调试使用技巧

快捷键shift+enter:Console中换行ctrl + shift + D (⌘ + shift + D Mac):切换控制台位置,右侧底部相互切换ctrl + [ 和 ctrl + ]:左右切换面板ctrl + 1 到 ctrl + 9:按索引切换面板(需Settings >>Preferences>>Appearance 打开此功能)ctrl + f:查找,任何面板都支持Ctrl + Shift + P (Mac: ⌘ + Shift+ P ):打开命令输入框,可搜索命令等Ctrl+P:搜索框,输入?可查看帮助控制台命令复制内容到剪切板:copy(…),当然你亦可在elements面板直接鼠标右键复制元素输出结果保存为全局变量:控制台右击 它,并选择 “Store as global variable” (保存为全局变量) 选项。保存日志:控制台右击选Save as保存日志截图:Elements面板,鼠标右击元素选Capture node screenshoot截图,或在命令输入框数入screenshoot搜索相关命令控制台布局:在命令输入框输入layout搜索切换主题:命令输入框输入theme切换dark、light主题切换页面保留控制台输出:Console面板->Console Setting->勾选Preserve log即可控制台脚本控制台支持直接运行js代码,复制回车运行即可,若想储存脚本到浏览器,命令输入框输入snippet打开Snippets面板进行添加/运行脚本,Ctrl+P搜索框键入!可快捷搜索Snippets。$0:对我们当前选中的 html 节点的引用。$1是对上一次我们选择的节点的引用,$2是对在那之前选择的节点的引用,等等。一直到$4$_:上次执行结果$i:安装npm插件,如$i('moment')$:类似jQuery元素选择器,如$('body')$$:类似$但它返回的是节点的 数组 ,等价于Array.from(document.querySelectorAll('div'))$x:xpath选择器,例如$x('/html/body/div[4]/div[2]/div[2]/div[2]/div[2]/div[2]/div')[0].textContent异步:console 默认就被 async 包裹,因此可以直接执行promiseres=await fetch('https://hu60.cn/q.php/index.index.json'); await res.json()queryObjects(x):对象查询,例如class Person { constructor(name, role) { this.name = name; this.role = role; } greet() { return this.getMessage('greeting'); } getMessage(type) { if (type === 'greeting') { return `Hello, I'm ${this.name}!`; } }}let john=new Person('John','kid')let kids=[new Person('Mary','kid'),new Person('Luke','kid')]new Person('tom','kid')queryObjects(Person)console.assert():console.assert() 方法在第一个参数为 false 的情况下会在控制台输出信息。console.assert(document.getElementById("demo"), "没有 ID 为 'demo' 的元素");console.table():将集合数组或对象以表格形式展示,例如console.table([{"name":"张三","age":18}]);执行耗时:如下,括号中定义定时器名称,可以留空console.time('timer1');console.timeEnd('timer1');Live Expression:Console界面点击眼睛图标,可给当前网站添加动态脚本,执行250ms一次。例如可以监控页面某元素是否变化监听事件:Elements中选择元素鼠标右键Store as gload variable,一般会保存为temp1类似变量名,使用monitorEvents(temp1); //开始记录unmonitorEvents(temp1); //停止记录monitorEvents(temp1, ['mouse', 'focus']); //记录指定事件Conditional breakpoints:在Sources里鼠标右键Add Conditional breakpoints或在断点上鼠标右击编辑,当条件符合才执行断点若条件为console.log则会直接执行,因此可以通过断点直接打印数据到控制台参考:https://www.frontendwingman.com/Chrome/

Chrome浏览器调试使用技巧

爬虫工具箱:InfoSpider

InfoSpider 一个集众多数据源于一身的爬虫工具箱,提供数据分析功能,基于用户数据生成图表文件,使得用户更直观、深入了解自己的信息。目前支持的数据源有:GitHub、QQ 邮箱、网易邮箱、阿里邮箱、新浪邮箱、Hotmail 邮箱、Outlook 邮箱、京东、淘宝、支付宝、中国移动、中国联通、中国电信、知乎、哔哩哔哩、网易云音乐、QQ 好友、QQ 群、生成朋友圈相册、浏览器浏览历史、12306、博客园、CSDN 博客、开源中国博客、简书。GitHub 地址→https://github.com/kangvcar/InfoSpider

Aid Learning 一款能在手机上运行的Linux系统

天生强大,Aid在Android上构架了一个支持图形化界面的完整Linux系统几乎支持所有深度学习开发框架:caffe/mxnet/keras/torch/tensorflow/ncnn/opencv…内置了强大的编程IDE和开发工具Aid Learning官网:http://www.aidlearning.net/

Aid Learning 一款能在手机上运行的Linux系统

VB使用记录知识点小结

以下教程适合VB6。Visual Basic 6.0是微软公司开发的编程设计软件,基于Windows操作系统可视化编程环境。Visual Basic 6.0还提供了窗口编辑,可直接对窗口进行编辑和预览。换行续写代码换行符:vbCrLf续写符:& _,若仅为语句结尾用下划线用就行续写符前面不能用+号拼接字符串若批量替换字符串换行符为vb支持方式,可使用支持正则替换文本编辑器执行替换:\r\n为" + vbCrLf & _\r\n+"代码格式化可使用在线工具:http://tools.jb51.net/code/vbscodeformat/执行shellshell直接执行Shell "ipconfig"WScript执行shellExec用Exec执行可读取执行结果 Dim result As String Dim WshShell Dim oExec Set WshShell = CreateObject("WScript.Shell") Set oExec = WshShell.Exec("ipconfig") result = oExec.StdOut.ReadAllRun以上执行命令都会打开黑窗口,可用Run隐藏,但其不能返回执行结果,只会返回成功失败Public Sub xshell(ByVal Ch As String) Dim WshShell Dim oExec Set WshShell = CreateObject("WScript.Shell") oExec = WshShell.Run(path + Ch, 0)End Sub具体使用详解:https://www.cnblogs.com/chulia20002001/p/6931903.html窗口跟随根据句柄获得某窗口左上坐标跟高宽,对跟随的form设置合适坐标即可,当然为实现实时跟随,定时器不能少。如下示例是使安卓三大按键贴在安卓模拟器窗口下面并跟随Option Explicit'根据句柄获取窗口Private Declare Function FindWindow Lib "user32" Alias "FindWindowA" ( _ ByVal lpClassName As String, _ ByVal lpWindowName As String) As Long '获取窗口范围 Private Declare Function GetWindowRect Lib "user32" _ (ByVal hwnd As Long, lpRect As RECT) As Long Private Type RECT Left As Long Top As Long Right As Long Bottom As Long End Type Dim NotepadHwnd As Long, NotepadRect As RECT Private Sub Form_Load() Me.Height = 450 Timer1.Interval = 100 Timer1.Enabled = True End Sub Private Sub Timer1_Timer() Dim oneWidth As Integer NotepadHwnd = FindWindow("SDL_app", vbNullString) '获取窗口句柄 If NotepadHwnd Then '如果成功 GetWindowRect NotepadHwnd, NotepadRect '获取窗口范围 Me.Left = (NotepadRect.Left + 7) * Screen.TwipsPerPixelX Me.Top = (NotepadRect.Bottom - 7) * Screen.TwipsPerPixelY '- Me.Height Me.Width = (NotepadRect.Right - NotepadRect.Left - 14) * Screen.TwipsPerPixelX oneWidth = Me.Width / 3 menu.Width = oneWidth menu.Height = Me.Height home.Left = oneWidth home.Width = oneWidth home.Height = Me.Height back.Left = Me.Width - oneWidth back.Width = oneWidth back.Height = Me.Height Else Form1.kqdhj.Value = 0 End If End Sub获取用户与程序目录当前Windows用户目录路径,例如下面就是Windows图片目录:Environ("userprofile") + "\Pictures\"当前程序所在目录:App.path文本框只允许输入数字以下是名为fps文本框只允许输入整数例子: Private Sub fps_KeyPress(KeyAscii As Integer) '只允许数字键,退格键,小数点进行输入的处理 Debug.Print KeyAscii Select Case KeyAscii Case vbKey0 To vbKey9, vbKeyBack '0 - 9,BACKSPACE处理 Case vbKeyDelete, vbKeyDecimal '小数点处理 If InStr(1, Text1.Text, ".") <> 0 Then KeyAscii = 0 Case Else KeyAscii = 0 End Select End Sub寻找字符串返回索引使用InStr,第一个参数从哪开始找,第二个源参数,第三个要寻找字符InStr(1, Text1.Text, ".")睡眠Sleep实现类似Java中的sleep函数Private Declare Sub Sleep Lib "kernel32" (ByVal dwMilliseconds As Long)'调用sleep 1000多选框选中事件value值1为选中 Private Sub istop_Click() If (istop.Value = 1) Then zd = " --always-on-top " Else zd = "" End If End Sub显示Winsock控件打开菜单:工程>部件或直接Ctrl+T呼出部件,勾选winsock即可获取本地IP地址使用Winsock控件获取Dim aa As StringDim strLocalIP As StringDim winIP As Objectaa = aa & "本机电脑名称:" & Environ("computername") & vbCrLfaa = aa & "本机用户名称:" & Environ("username") & vbCrLfSet winIP = CreateObject("MSWinsock.Winsock")strLocalIP = winIP.localipMsgBox aa & "本机IP:" & strLocalIP当然,当存在多个虚拟网卡时可能不精准,以下用命令行获取本地IP If (Mid(strLocalIP, 1, 3) <> "192") Then Dim result As String Dim WshShell Dim oExec Set WshShell = CreateObject("WScript.Shell") Set oExec = WshShell.Exec("ipconfig") result = oExec.StdOut.ReadAll Dim ex As Integer ex = InStr(1, result, "IPv4") If (ex > 0) Then Dim lines() As String Dim ipv4 As String lines() = Split(result, vbCrLf) '遍历数组 Dim i As Integer Dim s As String For i = 0 To UBound(lines) - 1 ex = InStr(1, lines(i), "IPv4") If (ex > 0) Then ex = InStr(1, lines(i), ":") ipv4 = Trim(Right(lines(i), Len(lines(i)) - ex)) If (Left(ipv4, 3) = "192") Then strLocalIP = ipv4 Label2.Caption = Label2.Caption + vbCrLf + ipv4 End If End If Next i End If End Ifqrmaker生成二维码使用第三方控件生成二维码,部件下载:http://www.opdown.com/soft/85169.html#download安装部件下载解压完成后将其中的Qrmaker.lic 、Qrmaker.tlb、Qrmaker.ocx 、Qrmaker.oca 和Regsvr32.exe文件拷贝到你新建的工程下,然后注册.ocx文件:打开“运行”,输入注册命令对于64位的系统。在“运行”中直接输入“regsvr32 文件路径\ Qrmaker.ocx”。例如“regsvr32 / qrmaker.ocx”出现“DllRegisterServer成功”消息确定,重启计算机即可。注册成功后再“工程”的“部件”控件中选择QRmaker.ocx控件即可。建议将使用非常简单,拖出一个部件代码修改InputData属性再刷新即可: QRmaker1.InputData = "我爱你" QRmaker1.Refresh

VB使用记录知识点小结

Python最简单HTTP服务器与MVC

socket模拟HTTP服务基于socket开发,监听127.0.0.1任一端口,如:8888,接收监听到的数据,并通过conn以HTTP响应的格式返回数据import socketsock = socket.socket()sock.bind(("localhost", 8888)) # 绑定监听的IP地址与端口8800sock.listen(5) # 设置最大监听数while True: conn, addr = sock.accept() data = conn.recv(1024) print(data) # 打印查看请求头与请求体 # 必须以HTTP响应头的格式返回数据,否则浏览器无法正常解析 # 同时注意send的数据不能是str字符串,必须是bytes,否则会报错。 conn.send(b'HTTP/1.1 200 OK\r\n\r\n <h1>Hello world!</h1>') conn.close()浏览器访问localhost:8800,即可看到网页结果,http response中响应体前必须有两空行’ \r\n\r\n’ 否则会被认为是响应头的内容https://blog.csdn.net/qq_29941979/article/details/107871763˂a name=利用Tornado库 href="#"˃利用Tornado库my.pyimport tornado.ioloopimport tornado.web#访问地址 http://127.0.0.1:9870/main?ywdm=06&num1=10&num2=200class TestClassA: def sub(self,a,b): return a-b def add(self,a,b): return a+b def chen(self,a,b): return a*bclass TestClassB: def sub(self,a,b): return a-b def add(self,a,b): return a+b def chen(self,a,b): return a*bsys_config={}sys_config["01"]=['mymvc','TestClassA','add']sys_config["02"]=['mymvc','TestClassA','sub']sys_config["03"]=['mymvc','TestClassA','chen']sys_config["04"]=['mymvc','TestClassB','add']sys_config["05"]=['mymvc','TestClassB','sub']sys_config["06"]=['mymvc','TestClassB','chen']class MainHandler(tornado.web.RequestHandler): def get(self): ywdm=self.get_argument('ywdm') num1=int(self.get_argument('num1').encode('utf-8')) num2=int(self.get_argument('num2').encode('utf-8')) message="hello !" print type(num2) if ywdm in sys_config: my_module_name=sys_config[ywdm][0] my_class_name=sys_config[ywdm][1] my_method_name=sys_config[ywdm][2] my_module = __import__(my_module_name) my_class = getattr(my_module,my_class_name) my_obj = my_class() my_method = getattr(my_obj,my_method_name) ret=my_method(num1,num2) print "ret:",ret #message = "ret data:::"+ret message="ywdm:"+ywdm+"-data:"+str(ret) items = ["Item 1", "Item 2", "Item 3"] self.render("test.html", title="My title", items=items,config_items=sys_config) #self.write(message) #self.finish()application = tornado.web.Application([(r"/main", MainHandler),])if __name__ == "__main__": application.listen(9870) tornado.ioloop.IOLoop.instance().start()test.html<html> <head> <title>{{ title }}</title> </head> <body> <ul> {% for item in items %} <li>{{ escape(item) }}</li> {% end %} </ul> <ul> {% for item in config_items %} <li>{{ escape(config_items[item][0]) }} -{{ escape(config_items[item][1]) }} -{{ escape(config_items[item][2]) }} </li> {% end %} </ul> </body> </html>https://blog.csdn.net/5iasp/article/details/23267609其中py中application中为数组形式,可以加多条映射,返回结果可以返回渲染后模板也可以返回字符串,例如:import tornado.ioloopimport tornado.webimport ctypesclass MainHandler(tornado.web.RequestHandler): def get(self): ip = self.get_argument('ip') print(ip) self.write("成功") self.finish()class IndexHandler(tornado.web.RequestHandler): def get(self): items = [] self.render("ip.html", title="My title", items=items)application = tornado.web.Application([(r"/index/get", MainHandler),(r"/index", IndexHandler) ])if __name__ == "__main__": # 最小化程序 ctypes.windll.user32.ShowWindow(ctypes.windll.kernel32.GetConsoleWindow(), 6) application.listen(9870) tornado.ioloop.IOLoop.instance().start()

安卓adb命令使用大全

基本用法### 命令语法adb 命令的基本语法如下:adb [-d|-e|-s <serialNumber>] <command>如果只有一个设备/模拟器连接时,可以省略掉 [-d|-e|-s <serialNumber>] 这一部分,直接使用 adb <command>。### 为命令指定目标设备如果有多个设备/模拟器连接,则需要为命令指定目标设备。参数含义-d指定当前唯一通过 USB 连接的 Android 设备为命令目标-e指定当前唯一运行的模拟器为命令目标-s <serialNumber>指定相应 serialNumber 号的设备/模拟器为命令目标在多个设备/模拟器连接的情况下较常用的是 -s <serialNumber> 参数,serialNumber 可以通过 adb devices 命令获取。如:$ adb devicesList of devices attachedcf264b8f deviceemulator-5554 device10.129.164.6:5555 device输出里的 cf264b8f、emulator-5554 和 10.129.164.6:5555 即为 serialNumber。比如这时想指定 cf264b8f 这个设备来运行 adb 命令获取屏幕分辨率:adb -s cf264b8f shell wm size又如想给 10.129.164.6:5555 这个设备安装应用(这种形式的 serialNumber 格式为 <IP>:<Port>,一般为无线连接的设备或 Genymotion 等第三方 Android 模拟器):adb -s 10.129.164.6:5555 install test.apk遇到多设备/模拟器的情况均使用这几个参数为命令指定目标设备,下文中为简化描述,不再重复。### 启动/停止启动 adb server 命令:adb start-server(一般无需手动执行此命令,在运行 adb 命令时若发现 adb server 没有启动会自动调起。)停止 adb server 命令:adb kill-server### 查看 adb 版本命令:adb version示例输出:Android Debug Bridge version 1.0.36Revision 8f855a3d9b35-android### 以 root 权限运行 adbdadb 的运行原理是 PC 端的 adb server 与手机端的守护进程 adbd 建立连接,然后 PC 端的 adb client 通过 adb server 转发命令,adbd 接收命令后解析运行。所以如果 adbd 以普通权限执行,有些需要 root 权限才能执行的命令无法直接用 adb xxx 执行。这时可以 adb shell 然后 su 后执行命令,也可以让 adbd 以 root 权限执行,这个就能随意执行高权限命令了。命令:adb root正常输出:restarting adbd as root现在再运行 adb shell,看看命令行提示符是不是变成 # 了?有些手机 root 后也无法通过 adb root 命令让 adbd 以 root 权限执行,比如三星的部分机型,会提示 adbd cannot run as root in production builds,此时可以先安装 adbd Insecure,然后 adb root 试试。相应地,如果要恢复 adbd 为非 root 权限的话,可以使用 adb unroot 命令。### 指定 adb server 的网络端口命令:adb -P <port> start-server默认端口为 5037。## 设备连接管理### 查询已连接设备/模拟器命令:adb devices输出示例:List of devices attachedcf264b8f deviceemulator-5554 device10.129.164.6:5555 device输出格式为 [serialNumber] [state],serialNumber 即我们常说的 SN,state 有如下几种:offline —— 表示设备未连接成功或无响应。device —— 设备已连接。注意这个状态并不能标识 Android 系统已经完全启动和可操作,在设备启动过程中设备实例就可连接到 adb,但启动完毕后系统才处于可操作状态。no device —— 没有设备/模拟器连接。以上输出显示当前已经连接了三台设备/模拟器,cf264b8f、emulator-5554 和 10.129.164.6:5555 分别是它们的 SN。从 emulator-5554 这个名字可以看出它是一个 Android 模拟器,而 10.129.164.6:5555 这种形为 <IP>:<Port> 的 serialNumber 一般是无线连接的设备或 Genymotion 等第三方 Android 模拟器。常见异常输出:没有设备/模拟器连接成功。List of devices attached设备/模拟器未连接到 adb 或无响应。List of devices attachedcf264b8f offline### USB 连接通过 USB 连接来正常使用 adb 需要保证几点:硬件状态正常。包括 Android 设备处于正常开机状态,USB 连接线和各种接口完好。Android 设备的开发者选项和 USB 调试模式已开启。可以到「设置」-「开发者选项」-「Android 调试」查看。如果在设置里找不到开发者选项,那需要通过一个彩蛋来让它显示出来:在「设置」-「关于手机」连续点击「版本号」7 次。设备驱动状态正常。这一点貌似在 Linux 和 Mac OS X 下不用操心,在 Windows 下有可能遇到需要安装驱动的情况,确认这一点可以右键「计算机」-「属性」,到「设备管理器」里查看相关设备上是否有黄色感叹号或问号,如果没有就说明驱动状态已经好了。否则可以下载一个手机助手类程序来安装驱动先。通过 USB 线连接好电脑和设备后确认状态。adb devices如果能看到xxxxxx device说明连接成功。### 无线连接(Android11 及以上)Android 11 及更高版本支持使用 Android 调试桥 (adb) 从工作站以无线方式部署和调试应用。例如,您可以将可调试应用部署到多台远程设备,而无需通过 USB 实际连接设备。这样就可以避免常见的 USB 连接问题,例如驱动程序安装方面的问题。官方文档操作步骤:更新到最新版本的 SDK 平台工具(至少30.0.0)。将 Android 设备与要运行 adb 的电脑连接到同一个局域网,比如连到同一个 WiFi。在开发者选项中启用无线调试。在询问要允许在此网络上进行无线调试吗?的对话框中,点击允许。选择使用配对码配对设备,使用弹窗中的 IP 地址和端口号。adb pair ipaddr:port提示Enter pairing code:时输入弹窗中的配对码,成功后会显示Successfully paired to ...。使用无线调试下的 IP 地址和端口。adb connect ipaddr:port确认连接状态。adb devices如果能看到ipaddr:port device说明连接成功。### 无线连接(需要借助 USB 线)除了可以通过 USB 连接设备与电脑来使用 adb,也可以通过无线连接——虽然连接过程中也有需要使用 USB 的步骤,但是连接成功之后你的设备就可以在一定范围内摆脱 USB 连接线的限制啦!操作步骤:将 Android 设备与要运行 adb 的电脑连接到同一个局域网,比如连到同一个 WiFi。将设备与电脑通过 USB 线连接。应确保连接成功(可运行 adb devices 看是否能列出该设备)。让设备在 5555 端口监听 TCP/IP 连接:adb tcpip 5555断开 USB 连接。找到设备的 IP 地址。一般能在「设置」-「关于手机」-「状态信息」-「IP地址」找到,也可以使用下文里 查看设备信息 - IP 地址 一节里的方法用 adb 命令来查看。通过 IP 地址连接设备。adb connect <device-ip-address>这里的 <device-ip-address> 就是上一步中找到的设备 IP 地址。确认连接状态。adb devices如果能看到<device-ip-address>:5555 device说明连接成功。如果连接不了,请确认 Android 设备与电脑是连接到了同一个 WiFi,然后再次执行 adb connect <device-ip-address> 那一步;如果还是不行的话,通过 adb kill-server 重新启动 adb 然后从头再来一次试试。断开无线连接命令:adb disconnect <device-ip-address>### 无线连接(无需借助 USB 线)注:需要 root 权限。上一节「无线连接(需要借助 USB 线)」是官方文档里介绍的方法,需要借助于 USB 数据线来实现无线连接。既然我们想要实现无线连接,那能不能所有步骤下来都是无线的呢?答案是能的。在 Android 设备上安装一个终端模拟器。已经安装过的设备可以跳过此步。我使用的终端模拟器下载地址是:Terminal Emulator for Android Downloads将 Android 设备与要运行 adb 的电脑连接到同一个局域网,比如连到同一个 WiFi。打开 Android 设备上的终端模拟器,在里面依次运行命令:susetprop service.adb.tcp.port 5555找到 Android 设备的 IP 地址。一般能在「设置」-「关于手机」-「状态信息」-「IP地址」找到,也可以使用下文里 查看设备信息 - IP 地址 一节里的方法用 adb 命令来查看。在电脑上通过 adb 和 IP 地址连接 Android 设备。adb connect <device-ip-address>这里的 <device-ip-address> 就是上一步中找到的设备 IP 地址。如果能看到 connected to <device-ip-address>:5555 这样的输出则表示连接成功。节注一:有的设备,比如小米 5S + MIUI 8.0 + Android 6.0.1 MXB48T,可能在第 5 步之前需要重启 adbd 服务,在设备的终端模拟器上运行:restart adbd如果 restart 无效,尝试以下命令:stop adbdstart adbd## 应用管理### 查看应用列表查看应用列表的基本命令格式是adb shell pm list packages [-f] [-d] [-e] [-s] [-3] [-i] [-u] [--user USER_ID] [FILTER]即在 adb shell pm list packages 的基础上可以加一些参数进行过滤查看不同的列表,支持的过滤参数如下:参数显示列表无所有应用-f显示应用关联的 apk 文件-d只显示 disabled 的应用-e只显示 enabled 的应用-s只显示系统应用-3只显示第三方应用-i显示应用的 installer-u包含已卸载应用<FILTER>包名包含 <FILTER> 字符串#### 所有应用命令:adb shell pm list packages输出示例:package:com.android.smoketestpackage:com.example.android.livecubespackage:com.android.providers.telephonypackage:com.google.android.googlequicksearchboxpackage:com.android.providers.calendarpackage:com.android.providers.mediapackage:com.android.protipspackage:com.android.documentsuipackage:com.android.gallerypackage:com.android.externalstorage...// other packages here...#### 系统应用命令:adb shell pm list packages -s#### 第三方应用命令:adb shell pm list packages -3#### 包名包含某字符串的应用比如要查看包名包含字符串 mazhuang 的应用列表,命令:adb shell pm list packages mazhuang当然也可以使用 grep 来过滤:adb shell pm list packages | grep mazhuang### 安装 APK命令格式:adb install [-lrtsdg] <path_to_apk>参数:adb install 后面可以跟一些可选参数来控制安装 APK 的行为,可用参数及含义如下:参数含义-l将应用安装到保护目录 /mnt/asec-r允许覆盖安装-t允许安装 AndroidManifest.xml 里 application 指定 android:testOnly="true" 的应用-s将应用安装到 sdcard-d允许降级覆盖安装-g授予所有运行时权限运行命令后如果见到类似如下输出(状态为 Success)代表安装成功:[100%] /data/local/tmp/1.apk pkg: /data/local/tmp/1.apkSuccess上面是当前最新版 v1.0.36 的 adb 的输出,会显示 push apk 文件到手机的进度百分比。使用旧版本 adb 的输出则是这样的:12040 KB/s (22205609 bytes in 1.801s) pkg: /data/local/tmp/SogouInput_android_v8.3_sweb.apkSuccess而如果状态为 Failure 则表示安装失败,比如:[100%] /data/local/tmp/map-20160831.apk pkg: /data/local/tmp/map-20160831.apkFailure [INSTALL_FAILED_ALREADY_EXISTS]常见安装失败输出代码、含义及可能的解决办法如下:输出含义解决办法INSTALL_FAILED_ALREADY_EXISTS应用已经存在,或卸载了但没卸载干净adb install 时使用 -r 参数,或者先 adb uninstall <packagename> 再安装INSTALL_FAILED_INVALID_APK无效的 APK 文件INSTALL_FAILED_INVALID_URI无效的 APK 文件名确保 APK 文件名里无中文INSTALL_FAILED_INSUFFICIENT_STORAGE空间不足清理空间INSTALL_FAILED_DUPLICATE_PACKAGE已经存在同名程序INSTALL_FAILED_NO_SHARED_USER请求的共享用户不存在INSTALL_FAILED_UPDATE_INCOMPATIBLE以前安装过同名应用,但卸载时数据没有移除;或者已安装该应用,但签名不一致先 adb uninstall <packagename> 再安装INSTALL_FAILED_SHARED_USER_INCOMPATIBLE请求的共享用户存在但签名不一致INSTALL_FAILED_MISSING_SHARED_LIBRARY安装包使用了设备上不可用的共享库INSTALL_FAILED_REPLACE_COULDNT_DELETE替换时无法删除INSTALL_FAILED_DEXOPTdex 优化验证失败或空间不足INSTALL_FAILED_OLDER_SDK设备系统版本低于应用要求INSTALL_FAILED_CONFLICTING_PROVIDER设备里已经存在与应用里同名的 content providerINSTALL_FAILED_NEWER_SDK设备系统版本高于应用要求INSTALL_FAILED_TEST_ONLY应用是 test-only 的,但安装时没有指定 -t 参数INSTALL_FAILED_CPU_ABI_INCOMPATIBLE包含不兼容设备 CPU 应用程序二进制接口的 native codeINSTALL_FAILED_MISSING_FEATURE应用使用了设备不可用的功能INSTALL_FAILED_CONTAINER_ERROR1. sdcard 访问失败; 2. 应用签名与 ROM 签名一致,被当作内置应用。1. 确认 sdcard 可用,或者安装到内置存储; 2. 打包时不与 ROM 使用相同签名。INSTALL_FAILED_INVALID_INSTALL_LOCATION1. 不能安装到指定位置; 2. 应用签名与 ROM 签名一致,被当作内置应用。1. 切换安装位置,添加或删除 -s 参数; 2. 打包时不与 ROM 使用相同签名。INSTALL_FAILED_MEDIA_UNAVAILABLE安装位置不可用一般为 sdcard,确认 sdcard 可用或安装到内置存储INSTALL_FAILED_VERIFICATION_TIMEOUT验证安装包超时INSTALL_FAILED_VERIFICATION_FAILURE验证安装包失败INSTALL_FAILED_PACKAGE_CHANGED应用与调用程序期望的不一致INSTALL_FAILED_UID_CHANGED以前安装过该应用,与本次分配的 UID 不一致清除以前安装过的残留文件INSTALL_FAILED_VERSION_DOWNGRADE已经安装了该应用更高版本使用 -d 参数INSTALL_FAILED_PERMISSION_MODEL_DOWNGRADE已安装 target SDK 支持运行时权限的同名应用,要安装的版本不支持运行时权限INSTALL_PARSE_FAILED_NOT_APK指定路径不是文件,或不是以 .apk 结尾INSTALL_PARSE_FAILED_BAD_MANIFEST无法解析的 AndroidManifest.xml 文件INSTALL_PARSE_FAILED_UNEXPECTED_EXCEPTION解析器遇到异常INSTALL_PARSE_FAILED_NO_CERTIFICATES安装包没有签名INSTALL_PARSE_FAILED_INCONSISTENT_CERTIFICATES已安装该应用,且签名与 APK 文件不一致先卸载设备上的该应用,再安装INSTALL_PARSE_FAILED_CERTIFICATE_ENCODING解析 APK 文件时遇到 CertificateEncodingExceptionINSTALL_PARSE_FAILED_BAD_PACKAGE_NAMEmanifest 文件里没有或者使用了无效的包名INSTALL_PARSE_FAILED_BAD_SHARED_USER_IDmanifest 文件里指定了无效的共享用户 IDINSTALL_PARSE_FAILED_MANIFEST_MALFORMED解析 manifest 文件时遇到结构性错误INSTALL_PARSE_FAILED_MANIFEST_EMPTY在 manifest 文件里找不到找可操作标签(instrumentation 或 application)INSTALL_FAILED_INTERNAL_ERROR因系统问题安装失败INSTALL_FAILED_USER_RESTRICTED用户被限制安装应用在开发者选项里将「USB安装」打开,如果已经打开了,那先关闭再打开。INSTALL_FAILED_DUPLICATE_PERMISSION应用尝试定义一个已经存在的权限名称INSTALL_FAILED_NO_MATCHING_ABIS应用包含设备的应用程序二进制接口不支持的 native codeINSTALL_CANCELED_BY_USER应用安装需要在设备上确认,但未操作设备或点了取消在设备上同意安装INSTALL_FAILED_ACWF_INCOMPATIBLE应用程序与设备不兼容INSTALL_FAILED_TEST_ONLYAPK 文件是使用 Android Studio 直接 RUN 编译出来的文件通过 Gradle 的 assembleDebug 或 assembleRelease 重新编译,或者 Generate Signed APKdoes not contain AndroidManifest.xml无效的 APK 文件is not a valid zip file无效的 APK 文件Offline设备未连接成功先将设备与 adb 连接成功unauthorized设备未授权允许调试error: device not found没有连接成功的设备先将设备与 adb 连接成功protocol failure设备已断开连接先将设备与 adb 连接成功Unknown option: -sAndroid 2.2 以下不支持安装到 sdcard不使用 -s 参数No space left on device空间不足清理空间Permission denied … sdcard …sdcard 不可用signatures do not match the previously installed version; ignoring!已安装该应用且签名不一致先卸载设备上的该应用,再安装参考:PackageManager.javaadb install 内部原理简介adb install 实际是分三步完成:push apk 文件到 /data/local/tmp。调用 pm install 安装。删除 /data/local/tmp 下的对应 apk 文件。所以,必要的时候也可以根据这个步骤,手动分步执行安装过程。### 卸载应用命令:adb uninstall [-k] <packagename><packagename> 表示应用的包名,-k 参数可选,表示卸载应用但保留数据和缓存目录。命令示例:adb uninstall com.qihoo360.mobilesafe表示卸载 360 手机卫士。### 清除应用数据与缓存命令:adb shell pm clear <packagename><packagename> 表示应用名包,这条命令的效果相当于在设置里的应用信息界面点击了「清除缓存」和「清除数据」。命令示例:adb shell pm clear com.qihoo360.mobilesafe表示清除 360 手机卫士的数据和缓存。### 查看前台 Activity命令:adb shell dumpsys activity activities | grep mResumedActivity输出示例:mResumedActivity: ActivityRecord{8079d7e u0 com.cyanogenmod.trebuchet/com.android.launcher3.Launcher t42}其中的 com.cyanogenmod.trebuchet/com.android.launcher3.Launcher 就是当前处于前台的 Activity。在 Windows 下以上命令可能不可用,可以尝试 adb shell dumpsys activity activities | findstr mResumedActivity 或 adb shell "dumpsys activity activities | grep mResumedActivity"。### 查看正在运行的 Services命令:adb shell dumpsys activity services [<packagename>]<packagename> 参数不是必须的,指定 <packagename> 表示查看与某个包名相关的 Services,不指定表示查看所有 Services。<packagename> 不一定要给出完整的包名,比如运行 adb shell dumpsys activity services org.mazhuang,那么包名 org.mazhuang.demo1、org.mazhuang.demo2 和 org.mazhuang123 等相关的 Services 都会列出来。### 查看应用详细信息命令:adb shell dumpsys package <packagename>输出中包含很多信息,包括 Activity Resolver Table、Registered ContentProviders、包名、userId、安装后的文件资源代码等路径、版本信息、权限信息和授予状态、签名版本信息等。<packagename> 表示应用包名。输出示例:Activity Resolver Table: Non-Data Actions: android.intent.action.MAIN: 5b4cba8 org.mazhuang.guanggoo/.SplashActivity filter 5ec9dcc Action: "android.intent.action.MAIN" Category: "android.intent.category.LAUNCHER" AutoVerify=falseRegistered ContentProviders: org.mazhuang.guanggoo/com.tencent.bugly.beta.utils.BuglyFileProvider: Provider{7a3c394 org.mazhuang.guanggoo/com.tencent.bugly.beta.utils.BuglyFileProvider}ContentProvider Authorities: [org.mazhuang.guanggoo.fileProvider]: Provider{7a3c394 org.mazhuang.guanggoo/com.tencent.bugly.beta.utils.BuglyFileProvider} applicationInfo=ApplicationInfo{7754242 org.mazhuang.guanggoo}Key Set Manager: [org.mazhuang.guanggoo] Signing KeySets: 501Packages: Package [org.mazhuang.guanggoo] (c1d7f): userId=10394 pkg=Package{55f714c org.mazhuang.guanggoo} codePath=/data/app/org.mazhuang.guanggoo-2 resourcePath=/data/app/org.mazhuang.guanggoo-2 legacyNativeLibraryDir=/data/app/org.mazhuang.guanggoo-2/lib primaryCpuAbi=null secondaryCpuAbi=null versionCode=74 minSdk=15 targetSdk=25 versionName=1.1.74 splits=[base] apkSigningVersion=2 applicationInfo=ApplicationInfo{7754242 org.mazhuang.guanggoo} flags=[ HAS_CODE ALLOW_CLEAR_USER_DATA ALLOW_BACKUP ] privateFlags=[ RESIZEABLE_ACTIVITIES ] dataDir=/data/user/0/org.mazhuang.guanggoo supportsScreens=[small, medium, large, xlarge, resizeable, anyDensity] timeStamp=2017-10-22 23:50:53 firstInstallTime=2017-10-22 23:50:25 lastUpdateTime=2017-10-22 23:50:55 installerPackageName=com.miui.packageinstaller signatures=PackageSignatures{af09595 [53c7caa2]} installPermissionsFixed=true installStatus=1 pkgFlags=[ HAS_CODE ALLOW_CLEAR_USER_DATA ALLOW_BACKUP ] requested permissions: android.permission.READ_PHONE_STATE android.permission.INTERNET android.permission.ACCESS_NETWORK_STATE android.permission.ACCESS_WIFI_STATE android.permission.READ_LOGS android.permission.WRITE_EXTERNAL_STORAGE android.permission.READ_EXTERNAL_STORAGE install permissions: android.permission.INTERNET: granted=true android.permission.ACCESS_NETWORK_STATE: granted=true android.permission.ACCESS_WIFI_STATE: granted=true User 0: ceDataInode=1155675 installed=true hidden=false suspended=false stopped=true notLaunched=false enabled=0 gids=[3003] runtime permissions: android.permission.READ_EXTERNAL_STORAGE: granted=true android.permission.READ_PHONE_STATE: granted=true android.permission.WRITE_EXTERNAL_STORAGE: granted=true User 999: ceDataInode=0 installed=false hidden=false suspended=false stopped=true notLaunched=true enabled=0 gids=[3003] runtime permissions:Dexopt state: [org.mazhuang.guanggoo] Instruction Set: arm64 path: /data/app/org.mazhuang.guanggoo-2/base.apk status: /data/app/org.mazhuang.guanggoo-2/oat/arm64/base.odex [compilation_filter=speed-profile, status=kOatUpToDa te]### 查看应用安装路径命令:adb shell pm path <PACKAGE>输出应用安装路径输出示例:adb shell pm path ecarx.weatherpackage:/data/app/ecarx.weather-1.apk## 与应用交互主要是使用 am <command> 命令,常用的 <command> 如下:command用途start [options] <INTENT>启动 <INTENT> 指定的 Activitystartservice [options] <INTENT>启动 <INTENT> 指定的 Servicebroadcast [options] <INTENT>发送 <INTENT> 指定的广播force-stop <packagename>停止 <packagename> 相关的进程<INTENT> 参数很灵活,和写 Android 程序时代码里的 Intent 相对应。用于决定 intent 对象的选项如下:参数含义-a <ACTION>指定 action,比如 android.intent.action.VIEW-c <CATEGORY>指定 category,比如 android.intent.category.APP_CONTACTS-n <COMPONENT>指定完整 component 名,用于明确指定启动哪个 Activity,如 com.example.app/.ExampleActivity<INTENT> 里还能带数据,就像写代码时的 Bundle 一样:参数含义--esn <EXTRA_KEY>null 值(只有 key 名)`-e–es <EXTRA_KEY> <EXTRA_STRING_VALUE>`--ez <EXTRA_KEY> <EXTRA_BOOLEAN_VALUE>boolean 值--ei <EXTRA_KEY> <EXTRA_INT_VALUE>integer 值--el <EXTRA_KEY> <EXTRA_LONG_VALUE>long 值--ef <EXTRA_KEY> <EXTRA_FLOAT_VALUE>float 值--eu <EXTRA_KEY> <EXTRA_URI_VALUE>URI--ecn <EXTRA_KEY> <EXTRA_COMPONENT_NAME_VALUE>component name--eia <EXTRA_KEY> <EXTRA_INT_VALUE>[,<EXTRA_INT_VALUE...]integer 数组--ela <EXTRA_KEY> <EXTRA_LONG_VALUE>[,<EXTRA_LONG_VALUE...]long 数组### 启动应用/ 调起 Activity指定Activity名称启动命令格式:adb shell am start [options] <INTENT>例如:adb shell am start -n com.tencent.mm/.ui.LauncherUI表示调起微信主界面。adb shell am start -n org.mazhuang.boottimemeasure/.MainActivity --es "toast" "hello, world"表示调起 org.mazhuang.boottimemeasure/.MainActivity 并传给它 string 数据键值对 toast - hello, world。不指定Activity名称启动(启动主Activity)命令格式:adb shell monkey -p <packagename> -c android.intent.category.LAUNCHER 1例如:adb shell monkey -p com.tencent.mm -c android.intent.category.LAUNCHER 1表示调起微信主界面。### 调起 Service命令格式:adb shell am startservice [options] <INTENT>例如:adb shell am startservice -n com.tencent.mm/.plugin.accountsync.model.AccountAuthenticatorService表示调起微信的某 Service。另外一个典型的用例是如果设备上原本应该显示虚拟按键但是没有显示,可以试试这个:adb shell am startservice -n com.android.systemui/.SystemUIService### 停止 Service命令格式:adb shell am stopservice [options] <INTENT>### 发送广播命令格式:adb shell am broadcast [options] <INTENT>可以向所有组件广播,也可以只向指定组件广播。例如,向所有组件广播 BOOT_COMPLETED:adb shell am broadcast -a android.intent.action.BOOT_COMPLETED又例如,只向 org.mazhuang.boottimemeasure/.BootCompletedReceiver 广播 BOOT_COMPLETED:adb shell am broadcast -a android.intent.action.BOOT_COMPLETED -n org.mazhuang.boottimemeasure/.BootCompletedReceiver这类用法在测试的时候很实用,比如某个广播的场景很难制造,可以考虑通过这种方式来发送广播。既能发送系统预定义的广播,也能发送自定义广播。如下是部分系统预定义广播及正常触发时机:action触发时机android.net.conn.CONNECTIVITY_CHANGE网络连接发生变化android.intent.action.SCREEN_ON屏幕点亮android.intent.action.SCREEN_OFF屏幕熄灭android.intent.action.BATTERY_LOW电量低,会弹出电量低提示框android.intent.action.BATTERY_OKAY电量恢复了android.intent.action.BOOT_COMPLETED设备启动完毕android.intent.action.DEVICE_STORAGE_LOW存储空间过低android.intent.action.DEVICE_STORAGE_OK存储空间恢复android.intent.action.PACKAGE_ADDED安装了新的应用android.net.wifi.STATE_CHANGEWiFi 连接状态发生变化android.net.wifi.WIFI_STATE_CHANGEDWiFi 状态变为启用/关闭/正在启动/正在关闭/未知android.intent.action.BATTERY_CHANGED电池电量发生变化android.intent.action.INPUT_METHOD_CHANGED系统输入法发生变化android.intent.action.ACTION_POWER_CONNECTED外部电源连接android.intent.action.ACTION_POWER_DISCONNECTED外部电源断开连接android.intent.action.DREAMING_STARTED系统开始休眠android.intent.action.DREAMING_STOPPED系统停止休眠android.intent.action.WALLPAPER_CHANGED壁纸发生变化android.intent.action.HEADSET_PLUG插入耳机android.intent.action.MEDIA_UNMOUNTED卸载外部介质android.intent.action.MEDIA_MOUNTED挂载外部介质android.os.action.POWER_SAVE_MODE_CHANGED省电模式开启(以上广播均可使用 adb 触发)### 强制停止应用命令:adb shell am force-stop <packagename>命令示例:adb shell am force-stop com.qihoo360.mobilesafe表示停止 360 安全卫士的一切进程与服务。### 收紧内存命令:adb shell am send-trim-memory <pid> <level>pid: 进程 ID level: HIDDEN、RUNNING_MODERATE、BACKGROUND、 RUNNING_LOW、MODERATE、RUNNING_CRITICAL、COMPLETE命令示例:adb shell am send-trim-memory 12345 RUNNING_LOW表示向 pid=12345 的进程,发出 level=RUNNING_LOW 的收紧内存命令。## 文件管理### 复制设备里的文件到电脑命令:adb pull <设备里的文件路径> [电脑上的目录]其中 电脑上的目录 参数可以省略,默认复制到当前目录。例:adb pull /sdcard/sr.mp4 ~/tmp/小技巧:设备上的文件路径可能需要 root 权限才能访问,如果你的设备已经 root 过,可以先使用 adb shell 和 su 命令在 adb shell 里获取 root 权限后,先 cp /path/on/device /sdcard/filename 将文件复制到 sdcard,然后 adb pull /sdcard/filename /path/on/pc。### 复制电脑里的文件到设备命令:adb push <电脑上的文件路径> <设备里的目录>例:adb push ~/sr.mp4 /sdcard/小技巧:设备上的文件路径普通权限可能无法直接写入,如果你的设备已经 root 过,可以先 adb push /path/on/pc /sdcard/filename,然后 adb shell 和 su 在 adb shell 里获取 root 权限后,cp /sdcard/filename /path/on/device。## 模拟按键/输入在 adb shell 里有个很实用的命令叫 input,通过它可以做一些有趣的事情。input 命令的完整 help 信息如下:Usage: input [<source>] <command> [<arg>...]The sources are: mouse keyboard joystick touchnavigation touchpad trackball stylus dpad gesture touchscreen gamepadThe commands and default sources are: text <string> (Default: touchscreen) keyevent [--longpress] <key code number or name> ... (Default: keyboard) tap <x> <y> (Default: touchscreen) swipe <x1> <y1> <x2> <y2> [duration(ms)] (Default: touchscreen) press (Default: trackball) roll <dx> <dy> (Default: trackball)比如使用 adb shell input keyevent <keycode> 命令,不同的 keycode 能实现不同的功能,完整的 keycode 列表详见 KeyEvent,摘引部分我觉得有意思的如下:keycode含义3HOME 键4返回键5打开拨号应用6挂断电话24增加音量25降低音量26电源键27拍照(需要在相机应用里)64打开浏览器82菜单键85播放/暂停86停止播放87播放下一首88播放上一首122移动光标到行首或列表顶部123移动光标到行末或列表底部126恢复播放127暂停播放164静音176打开系统设置187切换应用207打开联系人208打开日历209打开音乐210打开计算器220降低屏幕亮度221提高屏幕亮度223系统休眠224点亮屏幕231打开语音助手276如果没有 wakelock 则让系统休眠下面是 input 命令的一些用法举例。### 电源键命令:adb shell input keyevent 26执行效果相当于按电源键。### 菜单键命令:adb shell input keyevent 82### HOME 键命令:adb shell input keyevent 3### 返回键命令:adb shell input keyevent 4### 音量控制增加音量:adb shell input keyevent 24降低音量:adb shell input keyevent 25静音:adb shell input keyevent 164### 媒体控制播放/暂停:adb shell input keyevent 85停止播放:adb shell input keyevent 86播放下一首:adb shell input keyevent 87播放上一首:adb shell input keyevent 88恢复播放:adb shell input keyevent 126暂停播放:adb shell input keyevent 127### 点亮/熄灭屏幕可以通过上文讲述过的模拟电源键来切换点亮和熄灭屏幕,但如果明确地想要点亮或者熄灭屏幕,那可以使用如下方法。点亮屏幕:adb shell input keyevent 224熄灭屏幕:adb shell input keyevent 223### 滑动解锁如果锁屏没有密码,是通过滑动手势解锁,那么可以通过 input swipe 来解锁。命令(参数以机型 Nexus 5,向上滑动手势解锁举例):adb shell input swipe 300 1000 300 500参数 300 1000 300 500 分别表示起始点x坐标 起始点y坐标 结束点x坐标 结束点y坐标。### 输入文本在焦点处于某文本框时,可以通过 input 命令来输入文本。命令:adb shell input text hello现在 hello 出现在文本框了。## 查看日志Android 系统的日志分为两部分,底层的 Linux 内核日志输出到 /proc/kmsg,Android 的日志输出到 /dev/log。### Android 日志命令格式:[adb] logcat [<option>] ... [<filter-spec>] ...常用用法列举如下:#### 按级别过滤日志Android 的日志分为如下几个优先级(priority):V —— Verbose(最低,输出得最多)D —— DebugI —— InfoW —— WarningE —— ErrorF —— FatalS —— Silent(最高,啥也不输出)按某级别过滤日志则会将该级别及以上的日志输出。比如,命令:adb logcat *:W会将 Warning、Error、Fatal 和 Silent 日志输出。(注: 在 macOS 下需要给 *:W 这样以 * 作为 tag 的参数加双引号,如 adb logcat "*:W",不然会报错 no matches found: *:W。)#### 按 tag 和级别过滤日志<filter-spec> 可以由多个 <tag>[:priority] 组成。比如,命令:adb logcat ActivityManager:I MyApp:D *:S表示输出 tag ActivityManager 的 Info 以上级别日志,输出 tag MyApp 的 Debug 以上级别日志,及其它 tag 的 Silent 级别日志(即屏蔽其它 tag 日志)。#### 日志格式可以用 adb logcat -v <format> 选项指定日志输出格式。日志支持按以下几种 <format>:brief默认格式。格式为:<priority>/<tag>(<pid>): <message>示例:D/HeadsetStateMachine( 1785): Disconnected process message: 10, size: 0process格式为:<priority>(<pid>) <message>示例:D( 1785) Disconnected process message: 10, size: 0 (HeadsetStateMachine)tag格式为:<priority>/<tag>: <message>示例:D/HeadsetStateMachine: Disconnected process message: 10, size: 0raw格式为:<message>示例:Disconnected process message: 10, size: 0time格式为:<datetime> <priority>/<tag>(<pid>): <message>示例:08-28 22:39:39.974 D/HeadsetStateMachine( 1785): Disconnected process message: 10, size: 0threadtime格式为:<datetime> <pid> <tid> <priority> <tag>: <message>示例:08-28 22:39:39.974 1785 1832 D HeadsetStateMachine: Disconnected process message: 10, size: 0long格式为:[ <datetime> <pid>:<tid> <priority>/<tag> ]<message>示例:[ 08-28 22:39:39.974 1785: 1832 D/HeadsetStateMachine ]Disconnected process message: 10, size: 0指定格式可与上面的过滤同时使用。比如:adb logcat -v long ActivityManager:I *:S#### 清空日志adb logcat -c### 内核日志命令:adb shell dmesg输出示例:<6>[14201.684016] PM: noirq resume of devices complete after 0.982 msecs<6>[14201.685525] PM: early resume of devices complete after 0.838 msecs<6>[14201.753642] PM: resume of devices complete after 68.106 msecs<4>[14201.755954] Restarting tasks ... done.<6>[14201.771229] PM: suspend exit 2016-08-28 13:31:32.679217193 UTC<6>[14201.872373] PM: suspend entry 2016-08-28 13:31:32.780363596 UTC<6>[14201.872498] PM: Syncing filesystems ... done.中括号里的 [14201.684016] 代表内核开始启动后的时间,单位为秒。通过内核日志我们可以做一些事情,比如衡量内核启动时间,在系统启动完毕后的内核日志里找到 Freeing init memory 那一行前面的时间就是。## 查看设备信息### 型号命令:adb shell getprop ro.product.model输出示例:Nexus 5### 电池状况命令:adb shell dumpsys battery输入示例:Current Battery Service state: AC powered: false USB powered: true Wireless powered: false status: 2 health: 2 present: true level: 44 scale: 100 voltage: 3872 temperature: 280 technology: Li-poly其中 scale 代表最大电量,level 代表当前电量。上面的输出表示还剩下 44% 的电量。### 屏幕分辨率命令:adb shell wm size输出示例:Physical size: 1080x1920该设备屏幕分辨率为 1080px * 1920px。如果使用命令修改过,那输出可能是:Physical size: 1080x1920Override size: 480x1024表明设备的屏幕分辨率原本是 1080px * 1920px,当前被修改为 480px * 1024px。### 屏幕密度命令:adb shell wm density输出示例:Physical density: 420该设备屏幕密度为 420dpi。如果使用命令修改过,那输出可能是:Physical density: 480Override density: 160表明设备的屏幕密度原来是 480dpi,当前被修改为 160dpi。### 显示屏参数命令:adb shell dumpsys window displays输出示例:WINDOW MANAGER DISPLAY CONTENTS (dumpsys window displays) Display: mDisplayId=0 init=1080x1920 420dpi cur=1080x1920 app=1080x1794 rng=1080x1017-1810x1731 deferred=false layoutNeeded=false其中 mDisplayId 为 显示屏编号,init 是初始分辨率和屏幕密度,app 的高度比 init 里的要小,表示屏幕底部有虚拟按键,高度为 1920 - 1794 = 126px 合 42dp。### android_id命令:adb shell settings get secure android_id输出示例:51b6be48bac8c569### IMEI在 Android 4.4 及以下版本可通过如下命令获取 IMEI:adb shell dumpsys iphonesubinfo输出示例:Phone Subscriber Info: Phone Type = GSM Device ID = 860955027785041其中的 Device ID 就是 IMEI。而在 Android 5.0 及以上版本里这个命令输出为空,得通过其它方式获取了(需要 root 权限):adb shellsuservice call iphonesubinfo 1输出示例:Result: Parcel( 0x00000000: 00000000 0000000f 00360038 00390030 '........8.6.0.9.' 0x00000010: 00350035 00320030 00370037 00350038 '5.5.0.2.7.7.8.5.' 0x00000020: 00340030 00000031 '0.4.1... ')把里面的有效内容提取出来就是 IMEI 了,比如这里的是 860955027785041。参考:adb shell dumpsys iphonesubinfo not working since Android 5.0 Lollipop### Android 系统版本命令:adb shell getprop ro.build.version.release输出示例:5.0.2### IP 地址每次想知道设备的 IP 地址的时候都得「设置」-「关于手机」-「状态信息」-「IP地址」很烦对不对?通过 adb 可以方便地查看。命令:adb shell ifconfig | grep Mask输出示例:inet addr:10.130.245.230 Mask:255.255.255.252inet addr:127.0.0.1 Mask:255.0.0.0那么 10.130.245.230 就是设备 IP 地址。在有的设备上这个命令没有输出,如果设备连着 WiFi,可以使用如下命令来查看局域网 IP:adb shell ifconfig wlan0输出示例:wlan0: ip 10.129.160.99 mask 255.255.240.0 flags [up broadcast running multicast]或wlan0 Link encap:UNSPEC inet addr:10.129.168.57 Bcast:10.129.175.255 Mask:255.255.240.0 inet6 addr: fe80::66cc:2eff:fe68:b6b6/64 Scope: Link UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1 RX packets:496520 errors:0 dropped:0 overruns:0 frame:0 TX packets:68215 errors:0 dropped:0 overruns:0 carrier:0 collisions:0 txqueuelen:3000 RX bytes:116266821 TX bytes:8311736如果以上命令仍然不能得到期望的信息,那可以试试以下命令(部分系统版本里可用):adb shell netcfg输出示例:wlan0 UP 10.129.160.99/20 0x00001043 f8:a9:d0:17:42:4dlo UP 127.0.0.1/8 0x00000049 00:00:00:00:00:00p2p0 UP 0.0.0.0/0 0x00001003 fa:a9:d0:17:42:4dsit0 DOWN 0.0.0.0/0 0x00000080 00:00:00:00:00:00rmnet0 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rmnet1 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rmnet3 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rmnet2 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rmnet4 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rmnet6 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rmnet5 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rmnet7 DOWN 0.0.0.0/0 0x00000000 00:00:00:00:00:00rev_rmnet3 DOWN 0.0.0.0/0 0x00001002 4e:b7:e4:2e:17:58rev_rmnet2 DOWN 0.0.0.0/0 0x00001002 4e:f0:c8:bf:7a:cfrev_rmnet4 DOWN 0.0.0.0/0 0x00001002 a6:c0:3b:6b:c4:1frev_rmnet6 DOWN 0.0.0.0/0 0x00001002 66:bb:5d:64:2e:e9rev_rmnet5 DOWN 0.0.0.0/0 0x00001002 0e:1b:eb:b9:23:a0rev_rmnet7 DOWN 0.0.0.0/0 0x00001002 7a:d9:f6:81:40:5arev_rmnet8 DOWN 0.0.0.0/0 0x00001002 4e:e2:a9:bb:d0:1brev_rmnet0 DOWN 0.0.0.0/0 0x00001002 fe:65:d0:ca:82:a9rev_rmnet1 DOWN 0.0.0.0/0 0x00001002 da:d8:e8:4f:2e:fe可以看到网络连接名称、启用状态、IP 地址和 Mac 地址等信息。### Mac 地址命令:adb shell cat /sys/class/net/wlan0/address输出示例:f8:a9:d0:17:42:4d这查看的是局域网 Mac 地址,移动网络或其它连接的信息可以通过前面的小节「IP 地址」里提到的 adb shell netcfg 命令来查看。### CPU 信息命令:adb shell cat /proc/cpuinfo输出示例:Processor : ARMv7 Processor rev 0 (v7l)processor : 0BogoMIPS : 38.40processor : 1BogoMIPS : 38.40processor : 2BogoMIPS : 38.40processor : 3BogoMIPS : 38.40Features : swp half thumb fastmult vfp edsp neon vfpv3 tls vfpv4 idiva idivtCPU implementer : 0x51CPU architecture: 7CPU variant : 0x2CPU part : 0x06fCPU revision : 0Hardware : Qualcomm MSM 8974 HAMMERHEAD (Flattened Device Tree)Revision : 000bSerial : 0000000000000000这是 Nexus 5 的 CPU 信息,我们从输出里可以看到使用的硬件是 Qualcomm MSM 8974,processor 的编号是 0 到 3,所以它是四核的,采用的架构是 ARMv7 Processor rev 0 (v71)。### 内存信息命令:adb shell cat /proc/meminfo输出示例:MemTotal: 1027424 kBMemFree: 486564 kBBuffers: 15224 kBCached: 72464 kBSwapCached: 24152 kBActive: 110572 kBInactive: 259060 kBActive(anon): 79176 kBInactive(anon): 207736 kBActive(file): 31396 kBInactive(file): 51324 kBUnevictable: 3948 kBMlocked: 0 kBHighTotal: 409600 kBHighFree: 132612 kBLowTotal: 617824 kBLowFree: 353952 kBSwapTotal: 262140 kBSwapFree: 207572 kBDirty: 0 kBWriteback: 0 kBAnonPages: 265324 kBMapped: 47072 kBShmem: 1020 kBSlab: 57372 kBSReclaimable: 7692 kBSUnreclaim: 49680 kBKernelStack: 4512 kBPageTables: 5912 kBNFS_Unstable: 0 kBBounce: 0 kBWritebackTmp: 0 kBCommitLimit: 775852 kBCommitted_AS: 13520632 kBVmallocTotal: 385024 kBVmallocUsed: 61004 kBVmallocChunk: 209668 kB其中,MemTotal 就是设备的总内存,MemFree 是当前空闲内存。### 更多硬件与系统属性设备的更多硬件与系统属性可以通过如下命令查看:adb shell cat /system/build.prop这会输出很多信息,包括前面几个小节提到的「型号」和「Android 系统版本」等。输出里还包括一些其它有用的信息,它们也可通过 adb shell getprop <属性名> 命令单独查看,列举一部分属性如下:属性名含义ro.build.version.sdkSDK 版本ro.build.version.releaseAndroid 系统版本ro.build.version.security_patchAndroid 安全补丁程序级别ro.product.model型号ro.product.brand品牌ro.product.name设备名ro.product.board处理器型号ro.product.cpu.abilistCPU 支持的 abi 列表[节注一]persist.sys.isUsbOtgEnabled是否支持 OTGdalvik.vm.heapsize每个应用程序的内存上限ro.sf.lcd_density屏幕密度节注一:一些小厂定制的 ROM 可能修改过 CPU 支持的 abi 列表的属性名,如果用 ro.product.cpu.abilist 属性名查找不到,可以这样试试:adb shell cat /system/build.prop | grep ro.product.cpu.abi示例输出:ro.product.cpu.abi=armeabi-v7aro.product.cpu.abi2=armeabi## 修改设置注: 修改设置之后,运行恢复命令有可能显示仍然不太正常,可以运行 adb reboot 重启设备,或手动重启。修改设置的原理主要是通过 settings 命令修改 /data/data/com.android.providers.settings/databases/settings.db 里存放的设置值。### 分辨率命令:adb shell wm size 480x1024表示将分辨率修改为 480px * 1024px。恢复原分辨率命令:adb shell wm size reset### 屏幕密度命令:adb shell wm density 160表示将屏幕密度修改为 160dpi。恢复原屏幕密度命令:adb shell wm density reset### 显示区域命令:adb shell wm overscan 0,0,0,200四个数字分别表示距离左、上、右、下边缘的留白像素,以上命令表示将屏幕底部 200px 留白。恢复原显示区域命令:adb shell wm overscan reset### 关闭 USB 调试模式命令:adb shell settings put global adb_enabled 0恢复:用命令恢复不了了,毕竟关闭了 USB 调试 adb 就连接不上 Android 设备了。去设备上手动恢复吧:「设置」-「开发者选项」-「Android 调试」。### 允许/禁止访问非 SDK API允许访问非 SDK API:adb shell settings put global hidden_api_policy_pre_p_apps 1adb shell settings put global hidden_api_policy_p_apps 1禁止访问非 SDK API:adb shell settings delete global hidden_api_policy_pre_p_appsadb shell settings delete global hidden_api_policy_p_apps不需要设备获得 Root 权限。命令最后的数字的含义:值含义0禁止检测非 SDK 接口的调用。该情况下,日志记录功能被禁用,并且令 strict mode API,即 detectNonSdkApiUsage() 无效。不推荐。1仅警告——允许访问所有非 SDK 接口,但保留日志中的警告信息,可继续使用 strick mode API。2禁止调用深灰名单和黑名单中的接口。3禁止调用黑名单中的接口,但允许调用深灰名单中的接口。### 状态栏和导航栏的显示隐藏本节所说的相关设置对应 Cyanogenmod 里的「扩展桌面」。命令:adb shell settings put global policy_control <key-values><key-values> 可由如下几种键及其对应的值组成,格式为 <key1>=<value1>:<key2>=<value2>。key含义immersive.full同时隐藏immersive.status隐藏状态栏immersive.navigation隐藏导航栏immersive.preconfirms?这些键对应的值可则如下值用逗号组合:value含义apps所有应用*所有界面packagename指定应用-packagename排除指定应用例如:adb shell settings put global policy_control immersive.full=*表示设置在所有界面下都同时隐藏状态栏和导航栏。adb shell settings put global policy_control immersive.status=com.package1,com.package2:immersive.navigation=apps,-com.package3表示设置在包名为 com.package1 和 com.package2 的应用里隐藏状态栏,在除了包名为 com.package3 的所有应用里隐藏导航栏。## 实用功能### 屏幕截图截图保存到电脑:adb exec-out screencap -p > sc.png如果 adb 版本较老,无法使用 exec-out 命令,这时候建议更新 adb 版本。无法更新的话可以使用以下麻烦点的办法:先截图保存到设备里:adb shell screencap -p /sdcard/sc.png然后将 png 文件导出到电脑:adb pull /sdcard/sc.png可以使用 adb shell screencap -h 查看 screencap 命令的帮助信息,下面是两个有意义的参数及含义:参数含义-p指定保存文件为 png 格式-d display-id指定截图的显示屏编号(有多显示屏的情况下)实测如果指定文件名以 .png 结尾时可以省略 -p 参数;否则需要使用 -p 参数。如果不指定文件名,截图文件的内容将直接输出到 stdout。另外一种一行命令截图并保存到电脑的方法:Linux 和 Windowsadb shell screencap -p | sed "s/\r$//" > sc.pngMac OS Xadb shell screencap -p | gsed "s/\r$//" > sc.png这个方法需要用到 gnu sed 命令,在 Linux 下直接就有,在 Windows 下 Git 安装目录的 bin 文件夹下也有。如果确实找不到该命令,可以下载 sed for Windows 并将 sed.exe 所在文件夹添加到 PATH 环境变量里。而在 Mac 下使用系统自带的 sed 命令会报错:sed: RE error: illegal byte sequence需要安装 gnu-sed,然后使用 gsed 命令:brew install gnu-sed### 录制屏幕录制屏幕以 mp4 格式保存到 /sdcard:adb shell screenrecord /sdcard/filename.mp4需要停止时按 Ctrl-C,默认录制时间和最长录制时间都是 180 秒。如果需要导出到电脑:adb pull /sdcard/filename.mp4可以使用 adb shell screenrecord --help 查看 screenrecord 命令的帮助信息,下面是常见参数及含义:参数含义–size WIDTHxHEIGHT视频的尺寸,比如 1280x720,默认是屏幕分辨率。–bit-rate RATE视频的比特率,默认是 4Mbps。–time-limit TIME录制时长,单位秒。–verbose输出更多信息。### 重新挂载 system 分区为可写注:需要 root 权限。/system 分区默认挂载为只读,但有些操作比如给 Android 系统添加命令、删除自带应用等需要对 /system 进行写操作,所以需要重新挂载它为可读写。步骤:进入 shell 并切换到 root 用户权限。命令:adb shellsu查看当前分区挂载情况。命令:mount输出示例:rootfs / rootfs ro,relatime 0 0tmpfs /dev tmpfs rw,seclabel,nosuid,relatime,mode=755 0 0devpts /dev/pts devpts rw,seclabel,relatime,mode=600 0 0proc /proc proc rw,relatime 0 0sysfs /sys sysfs rw,seclabel,relatime 0 0selinuxfs /sys/fs/selinux selinuxfs rw,relatime 0 0debugfs /sys/kernel/debug debugfs rw,relatime 0 0none /var tmpfs rw,seclabel,relatime,mode=770,gid=1000 0 0none /acct cgroup rw,relatime,cpuacct 0 0none /sys/fs/cgroup tmpfs rw,seclabel,relatime,mode=750,gid=1000 0 0none /sys/fs/cgroup/memory cgroup rw,relatime,memory 0 0tmpfs /mnt/asec tmpfs rw,seclabel,relatime,mode=755,gid=1000 0 0tmpfs /mnt/obb tmpfs rw,seclabel,relatime,mode=755,gid=1000 0 0none /dev/memcg cgroup rw,relatime,memory 0 0none /dev/cpuctl cgroup rw,relatime,cpu 0 0none /sys/fs/cgroup tmpfs rw,seclabel,relatime,mode=750,gid=1000 0 0none /sys/fs/cgroup/memory cgroup rw,relatime,memory 0 0none /sys/fs/cgroup/freezer cgroup rw,relatime,freezer 0 0/dev/block/platform/msm_sdcc.1/by-name/system /system ext4 ro,seclabel,relatime,data=ordered 0 0/dev/block/platform/msm_sdcc.1/by-name/userdata /data ext4 rw,seclabel,nosuid,nodev,relatime,noauto_da_alloc,data=ordered 0 0/dev/block/platform/msm_sdcc.1/by-name/cache /cache ext4 rw,seclabel,nosuid,nodev,relatime,data=ordered 0 0/dev/block/platform/msm_sdcc.1/by-name/persist /persist ext4 rw,seclabel,nosuid,nodev,relatime,data=ordered 0 0/dev/block/platform/msm_sdcc.1/by-name/modem /firmware vfat ro,context=u:object_r:firmware_file:s0,relatime,uid=1000,gid=1000,fmask=0337,dmask=0227,codepage=cp437,iocharset=iso8859-1,shortname=lower,errors=remount-ro 0 0/dev/fuse /mnt/shell/emulated fuse rw,nosuid,nodev,relatime,user_id=1023,group_id=1023,default_permissions,allow_other 0 0/dev/fuse /mnt/shell/emulated/0 fuse rw,nosuid,nodev,relatime,user_id=1023,group_id=1023,default_permissions,allow_other 0 0找到其中我们关注的带 /system 的那一行:/dev/block/platform/msm_sdcc.1/by-name/system /system ext4 ro,seclabel,relatime,data=ordered 0 0重新挂载。命令:mount -o remount,rw -t yaffs2 /dev/block/platform/msm_sdcc.1/by-name/system /system这里的 /dev/block/platform/msm_sdcc.1/by-name/system 就是我们从上一步的输出里得到的文件路径。如果输出没有提示错误的话,操作就成功了,可以对 /system 下的文件为所欲为了。### 查看连接过的 WiFi 密码注:需要 root 权限。命令:adb shellsucat /data/misc/wifi/*.conf输出示例:network={ ssid="TP-LINK_9DFC" scan_ssid=1 psk="123456789" key_mgmt=WPA-PSK group=CCMP TKIP auth_alg=OPEN sim_num=1 priority=13893}network={ ssid="TP-LINK_F11E" psk="987654321" key_mgmt=WPA-PSK sim_num=1 priority=17293}ssid 即为我们在 WLAN 设置里看到的名称,psk 为密码,key_mgmt 为安全加密方式。如果 Android O 或以后,WiFi 密码保存的地址有变化,是在 WifiConfigStore.xml 里面adb shellsucat /data/misc/wifi/WifiConfigStore.xml输出格式:数据项较多,只需关注 ConfigKey(WiFi 名字)和 PreSharedKey(WiFi 密码)即可<string name="ConfigKey">&quot;Wi-Fi&quot;WPA_PSK</string><string name="PreSharedKey">&quot;931907334&quot;</string>### 设置系统日期和时间注:需要 root 权限。命令:adb shellsudate -s 20160823.131500表示将系统日期和时间更改为 2016 年 08 月 23 日 13 点 15 分 00 秒。### 重启手机命令:adb reboot### 检测设备是否已 root命令:adb shellsu此时命令行提示符是 $ 则表示没有 root 权限,是 # 则表示已 root。### 使用 Monkey 进行压力测试Monkey 可以生成伪随机用户事件来模拟单击、触摸、手势等操作,可以对正在开发中的程序进行随机压力测试。简单用法:adb shell monkey -p <packagename> -v 500表示向 <packagename> 指定的应用程序发送 500 个伪随机事件。Monkey 的详细用法参考 官方文档。### 开启/关闭 WiFi注:需要 root 权限。有时需要控制设备的 WiFi 状态,可以用以下指令完成。开启 WiFi:adb rootadb shell svc wifi enable关闭 WiFi:adb rootadb shell svc wifi disable若执行成功,输出为空;若未取得 root 权限执行此命令,将执行失败,输出 Killed。## 刷机相关命令### 重启到 Recovery 模式命令:adb reboot recovery### 从 Recovery 重启到 Android命令:adb reboot### 重启到 Fastboot 模式命令:adb reboot bootloader### 通过 sideload 更新系统如果我们下载了 Android 设备对应的系统更新包到电脑上,那么也可以通过 adb 来完成更新。以 Recovery 模式下更新为例:重启到 Recovery 模式。命令:adb reboot recovery在设备的 Recovery 界面上操作进入 Apply update-Apply from ADB。注:不同的 Recovery 菜单可能与此有差异,有的是一级菜单就有 Apply update from ADB。通过 adb 上传和更新系统。命令:adb sideload <path-to-update.zip>## 安全相关命令### 启用/禁用 SELinux启用 SELinuxadb rootadb shell setenforce 1禁用 SELinuxadb rootadb shell setenforce 0### 启用/禁用 dm_verity启用 dm_verityadb rootadb enable-verity禁用 dm_verityadb rootadb disable-verity## 更多 adb shell 命令Android 系统是基于 Linux 内核的,所以 Linux 里的很多命令在 Android 里也有相同或类似的实现,在 adb shell 里可以调用。本文档前面的部分内容已经用到了 adb shell 命令。### 查看进程命令:adb shell ps输出示例:USER PID PPID VSIZE RSS WCHAN PC NAMEroot 1 0 8904 788 ffffffff 00000000 S /initroot 2 0 0 0 ffffffff 00000000 S kthreadd...u0_a71 7779 5926 1538748 48896 ffffffff 00000000 S com.sohu.inputmethod.sogou:classicu0_a58 7963 5926 1561916 59568 ffffffff 00000000 S org.mazhuang.boottimemeasure...shell 8750 217 10640 740 00000000 b6f28340 R ps各列含义:列名含义USER所属用户PID进程 IDPPID父进程 IDNAME进程名### 查看实时资源占用情况命令:adb shell top输出示例:User 0%, System 6%, IOW 0%, IRQ 0%User 3 + Nice 0 + Sys 21 + Idle 280 + IOW 0 + IRQ 0 + SIRQ 3 = 307 PID PR CPU% S #THR VSS RSS PCY UID Name 8763 0 3% R 1 10640K 1064K fg shell top 131 0 3% S 1 0K 0K fg root dhd_dpc 6144 0 0% S 115 1682004K 115916K fg system system_server 132 0 0% S 1 0K 0K fg root dhd_rxf 1731 0 0% S 6 20288K 788K fg root /system/bin/mpdecision 217 0 0% S 6 18008K 356K fg shell /sbin/adbd ... 7779 2 0% S 19 1538748K 48896K bg u0_a71 com.sohu.inputmethod.sogou:classic 7963 0 0% S 18 1561916K 59568K fg u0_a58 org.mazhuang.boottimemeasure ...各列含义:列名含义PID进程 IDPR优先级CPU%当前瞬间占用 CPU 百分比S进程状态(R=运行,S=睡眠,T=跟踪/停止,Z=僵尸进程)#THR线程数VSSVirtual Set Size 虚拟耗用内存(包含共享库占用的内存)RSSResident Set Size 实际使用物理内存(包含共享库占用的内存)PCY调度策略优先级,SP_BACKGROUND/SPFOREGROUNDUID进程所有者的用户 IDNAME进程名top 命令还支持一些命令行参数,详细用法如下:Usage: top [ -m max_procs ] [ -n iterations ] [ -d delay ] [ -s sort_column ] [ -t ] [ -h ] -m num 最多显示多少个进程 -n num 刷新多少次后退出 -d num 刷新时间间隔(单位秒,默认值 5) -s col 按某列排序(可用 col 值:cpu, vss, rss, thr) -t 显示线程信息 -h 显示帮助文档### 查看进程 UID有两种方案:adb shell dumpsys package <packagename> | grep userId=如:$ adb shell dumpsys package org.mazhuang.guanggoo | grep userId= userId=10394通过 ps 命令找到对应进程的 pid 之后 adb shell cat /proc/<pid>/status | grep Uid如:$ adb shellgemini:/ $ ps | grep org.mazhuang.guanggoou0_a394 28635 770 1795812 78736 SyS_epoll_ 0000000000 S org.mazhuang.guanggoogemini:/ $ cat /proc/28635/status | grep UidUid: 10394 10394 10394 10394gemini:/ $### 其它如下是其它常用命令的简单描述,前文已经专门讲过的命令不再额外说明:命令功能cat显示文件内容cd切换目录chmod改变文件的存取模式/访问权限df查看磁盘空间使用情况grep过滤输出kill杀死指定 PID 的进程ls列举目录内容mount挂载目录的查看和管理mv移动或重命名文件ps查看正在运行的进程rm删除文件top查看进程的资源占用情况## 常见问题### 启动 adb server 失败出错提示error: protocol fault (couldn't read status): No error可能原因adb server 进程想使用的 5037 端口被占用。解决方案找到占用 5037 端口的进程,然后终止它。以 Windows 下为例:netstat -ano | findstr LISTENING...TCP 0.0.0.0:5037 0.0.0.0:0 LISTENING 1548...这里 1548 即为进程 ID,用命令结束该进程:taskkill /PID 1548然后再启动 adb 就没问题了。### com.android.ddmlib.AdbCommandRejectedException在 Android Studio 里新建一个模拟器,但是用 adb 一直连接不上,提示:com.android.ddmlib.AdbCommandRejectedException: device unauthorized.This adb server's $ADB_VENDOR_KEYS is not setTry 'adb kill-server' if that seems wrong.Otherwise check for a confirmation dialog on your device.在手机上安装一个终端然后执行 su 提示没有该命令,这不正常。于是删除该模拟器后重新下载安装一次,这次就正常了。## adb 的非官方实现fb-adb - A better shell for Android devices (for Mac).## 相关命令aaptamdumsyspmuiautomator## 致谢感谢朋友们无私的分享与补充(排名不分先后)。zxning,linhua55,codeskyblue,seasonyuu,fan123199,zhEdward,0x8BADFOOD,keith666666,shawnlinboy,s-xq, lucky9322。## 参考链接Android Debug BridgeADB Shell Commandslogcat Command-line ToolAndroid ADB命令大全[adb 命令行的使用记录](https://github.com/ZQiang94/StudyRecords/blob/master/other/src/main/java/com/other/adb 命令行的使用记录.md)Android ADB命令大全(通过ADB命令查看wifi密码、MAC地址、设备信息、操作文件、查看文件、日志信息、卸载、启动和安装APK等)那些做Android开发必须知道的ADB命令adb shell top像高手一样使用ADB命令行(2)

纯HTML/CSS节日灯笼代码 给你网站加点节日气息

效果图:灯笼可自适应网页大小,并且自动贴两边代码: <style type="text/css"> .ct2 .mn {width:770px;} .ct2 .sd {width:218px;} @media screen and (max-width:768px) {.xnkl {display:none;} } .deng-box {position:fixed;top:-40px;right:150px;z-index:9999;pointer-events:none;} .deng-box1 {position:fixed;top:-30px;right:10px;z-index:9999;pointer-events:none} .deng-box2 {position:fixed;top:-40px;left:150px;z-index:9999;pointer-events:none} .deng-box3 {position:fixed;top:-30px;left:10px;z-index:9999;pointer-events:none} .deng-box1 .deng,.deng-box3 .deng {position:relative;width:120px;height:90px;margin:50px;background:#d8000f;background:rgba(216,0,15,.8);border-radius:50% 50%;-webkit-transform-origin:50% -100px;-webkit-animation:swing 5s infinite ease-in-out;box-shadow:-5px 5px 30px 4px #fc903d} .deng {position:relative;width:120px;height:90px;margin:50px;background:#d8000f;background:rgba(216,0,15,.8);border-radius:50% 50%;-webkit-transform-origin:50% -100px;-webkit-animation:swing 3s infinite ease-in-out;box-shadow:-5px 5px 50px 4px #fa6c00} .deng-a {width:100px;height:90px;background:#d8000f;background:rgba(216,0,15,.1);margin:12px 8px 8px 8px;border-radius:50% 50%;border:2px solid #dc8f03} .deng-b {width:45px;height:90px;background:#d8000f;background:rgba(216,0,15,.1);margin:-4px 8px 8px 26px;border-radius:50% 50%;border:2px solid #dc8f03} .xian {position:absolute;top:-20px;left:60px;width:2px;height:20px;background:#dc8f03} .shui-a {position:relative;width:5px;height:20px;margin:-5px 0 0 59px;-webkit-animation:swing 4s infinite ease-in-out;-webkit-transform-origin:50% -45px;background:orange;border-radius:0 0 5px 5px} .shui-b {position:absolute;top:14px;left:-2px;width:10px;height:10px;background:#dc8f03;border-radius:50%} .shui-c {position:absolute;top:18px;left:-2px;width:10px;height:35px;background:orange;border-radius:0 0 0 5px} .deng:before {position:absolute;top:-7px;left:29px;height:12px;width:60px;content:" ";display:block;z-index:999;border-radius:5px 5px 0 0;border:solid 1px #dc8f03;background:orange;background:linear-gradient(to right,#dc8f03,orange,#dc8f03,orange,#dc8f03)} .deng:after {position:absolute;bottom:-7px;left:10px;height:12px;width:60px;content:" ";display:block;margin-left:20px;border-radius:0 0 5px 5px;border:solid 1px #dc8f03;background:orange;background:linear-gradient(to right,#dc8f03,orange,#dc8f03,orange,#dc8f03)} .deng-t {font-family:黑体,Arial,Lucida Grande,Tahoma,sans-serif;font-size:3.2rem;color:#dc8f03;font-weight:700;line-height:85px;text-align:center} .night .deng-box,.night .deng-box1,.night .deng-t {background:0 0!important} @-moz-keyframes swing {0% {-moz-transform:rotate(-10deg)} 50% {-moz-transform:rotate(10deg)} 100% {-moz-transform:rotate(-10deg)} } @-webkit-keyframes swing {0% {-webkit-transform:rotate(-10deg)} 50% {-webkit-transform:rotate(10deg)} 100% {-webkit-transform:rotate(-10deg)} } </style> <div id="wp"class="wp"><div class="xnkl"><div class="deng-box2"><div class="deng"><div class="xian"></div><div class="deng-a"><div class="deng-b"><div class="deng-t">度</div></div></div><div class="shui shui-a"><div class="shui-c"></div><div class="shui-b"></div></div></div></div><div class="deng-box3"><div class="deng"><div class="xian"></div><div class="deng-a"><div class="deng-b"><div class="deng-t">欢</div></div></div><div class="shui shui-a"><div class="shui-c"></div><div class="shui-b"></div></div></div></div><div class="deng-box1"><div class="deng"><div class="xian"></div><div class="deng-a"><div class="deng-b"><div class="deng-t">春</div></div></div><div class="shui shui-a"><div class="shui-c"></div><div class="shui-b"></div></div></div></div><div class="deng-box"><div class="deng"><div class="xian"></div><div class="deng-a"><div class="deng-b"><div class="deng-t">新</div></div></div><div class="shui shui-a"><div class="shui-c"></div><div class="shui-b"></div></div></div></div></div>

纯HTML/CSS节日灯笼代码 给你网站加点节日气息

Win10电脑变蓝牙音箱教程

微软win10 2004版本增加了A2DP SNK功能,该教程请确保系统在此系统版本以上,win+r命令行输入:winver在弹出窗口确定系统版本,版本不满足请自行更新系统。首先蓝牙配对连接手机跟电脑,然后,打开Windows应用商店,搜索Bluetooth Audio Receiver下载安装即可,打开软件选择我手机的蓝牙名称,点击Open Connection,等待它连接完成即可。此教程可配Scrcpy完成电脑有声投屏幕。

Scrcpy相关文件下载汇总

Scrcpy相关资源可至开源项目下载,当然开源项目托管在github上下载太慢,因此我将部分Scrcpy资源放在网盘供下载,windows电脑根据自己的系统版本(64还是32位的选择合适版本),ScrcpyWin为通用版本,64、32位都可以使用。下载地址:https://hik.lanzous.com/b0107qicf密码:2gij

ScrcpyWin便捷启动工具v1.0.1

简介ScrcpyWin是一款Windows上可视化启动Scrcpy的GUI工具。可让你更便携地控制手机,绿色便携一触即发。ScrcpyWin单文件下载:v1.0.1 版:https://hik.lanzous.com/iIkOtm0desh更新日志:优化细节加入快捷键说明,加入安卓导航键开关v1.0 版本:https://hik.lanzous.com/iYIVFlreqziScrcpy 是一款免费开源的投屏软件,支持将安卓手机屏幕投放在 Windows、macOS、GNU/Linux 上,并可直接借助鼠标在投屏窗口中进行交互和录制。项目地址:https://github.com/Genymobile/scrcpy网盘下载:https://cway.top/post/889.html使用说明手机端准备手机端请在设置中连点手机版本号打开开发者模式,在开发者模式中开启USB调试与USB调试(安全设置),开启安全设置主要是方便可以在电脑端控制手机屏幕。电脑端使用解压scrcpy-win64-v1.17.zip到文件夹,下载ScrcpyWin_1.0.exe移动至scrcpy程序目录即可(必须放程序目录中,否则启动会提示找不到文件),启动ScrcpyWin_1.0.exe即可愉快使用,当然你可以将该exe发送快捷方式到桌面。用数据线连接电脑,打开软件后看到如下界面:刷新设备右侧框中显示设备device名称说明设备已连接,拔出或插上手机都请点击刷新设备,有设备代表可以连接,请选择其中一个设备开始投屏,若只有一个设备可不选择。无线连接请输入正确手机局域网IP(手机与电脑必须同一局域网),小米手机可在 设置-我的设备-全部参数-状态信息 中查看本机局域网IP地址,其它品牌手机可扫软件右下角二维码获取app查看或者使用其它方式查看。注意:无线网第一次连接请插上数据线激活,输入正确无线地址后点无线连接,连接后刷新设备会看到两个设备,一个随机字符串开头为有线连接设备,一个IP开头的设备即为无线设备,此时拔掉数据线即可。之后通过断开连接可直接通过无线连接按钮进行连接。投屏快捷键在以下列表中, MOD 是快捷键的修饰键。 默认是 (左) Alt 或 (左) Super。Windows上默认MOD键为Alt,以下快捷键请用alt代替。操作快捷键全屏MOD+f向左旋转屏幕MOD+← (左箭头)向右旋转屏幕MOD+→ (右箭头)将窗口大小重置为1:1 (匹配像素)MOD+g将窗口大小重置为消除黑边MOD+w | 双击¹点按 主屏幕MOD+h | 鼠标中键点按 返回MOD+b | 鼠标右键²点按 切换应用MOD+s点按 菜单 (解锁屏幕)MOD+m点按 音量+MOD+↑ (上箭头)点按 音量-MOD+↓ (下箭头)点按 电源MOD+p打开屏幕鼠标右键²关闭设备屏幕 (但继续在电脑上显示)MOD+o打开设备屏幕MOD+Shift+o旋转设备屏幕MOD+r展开通知面板MOD+n收起通知面板MOD+Shift+n复制到剪贴板³MOD+c剪切到剪贴板³MOD+x同步剪贴板并粘贴³MOD+v注入电脑剪贴板文本MOD+Shift+v打开/关闭FPS显示 (在 stdout)MOD+i捏拉缩放Ctrl+按住并移动鼠标¹双击黑边可以去除黑边 ²点击鼠标右键将在屏幕熄灭时点亮屏幕,其余情况则视为按下返回键 。 ³需要安卓版本 Android >= 7。所有的 Ctrl+按键 的快捷键都会被转发到设备,所以会由当前应用程序进行处理。详细说明请访问项目:https://github.com/Genymobile/scrcpy/blob/master/README.zh-Hans.md

ScrcpyWin便捷启动工具v1.0.1