python执行adb命令_Python实现对adb命令封装

我就废话不多说了,大家还是直接看代码吧!

#!/usr/bin/evn python

# -*- coding:utf-8 -*-

# FileName adbtools.py

# Author: HeyNiu

# Created Time: 2016/9/19

"""

adb 工具类

"""

import os

import platform

import re

import time

import utils.timetools

class AdbTools(object):

def __init__(self, device_id=''):

self.__system = platform.system()

self.__find = ''

self.__command = ''

self.__device_id = device_id

self.__get_find()

self.__check_adb()

self.__connection_devices()

def __get_find(self):

"""

判断系统类型,windows使用findstr,linux使用grep

:return:

"""

if self.__system is "Windows":

self.__find = "findstr"

else:

self.__find = "grep"

def __check_adb(self):

"""

检查adb

判断是否设置环境变量ANDROID_HOME

:return:

"""

if "ANDROID_HOME" in os.environ:

if self.__system == "Windows":

path = os.path.join(os.environ["ANDROID_HOME"], "platform-tools", "adb.exe")

if os.path.exists(path):

self.__command = path

else:

raise EnvironmentError(

"Adb not found in $ANDROID_HOME path: %s." % os.environ["ANDROID_HOME"])

else:

path = os.path.join(os.environ["ANDROID_HOME"], "platform-tools", "adb")

if os.path.exists(path):

self.__command = path

else:

raise EnvironmentError(

"Adb not found in $ANDROID_HOME path: %s." % os.environ["ANDROID_HOME"])

else:

raise EnvironmentError(

"Adb not found in $ANDROID_HOME path: %s." % os.environ["ANDROID_HOME"])

def __connection_devices(self):

"""

连接指定设备,单个设备可不传device_id

:return:

"""

if self.__device_id == "":

return

self.__device_id = "-s %s" % self.__device_id

def adb(self, args):

"""

执行adb命令

:param args:参数

:return:

"""

cmd = "%s %s %s" % (self.__command, self.__device_id, str(args))

# print(cmd)

return os.popen(cmd)

def shell(self, args):

"""

执行adb shell命令

:param args:参数

:return:

"""

cmd = "%s %s shell %s" % (self.__command, self.__device_id, str(args))

# print(cmd)

return os.popen(cmd)

def mkdir(self, path):

"""

创建目录

:param path: 路径

:return:

"""

return self.shell('mkdir %s' % path)

def get_devices(self):

"""

获取设备列表

:return:

"""

l = self.adb('devices').readlines()

return (i.split()[0] for i in l if 'devices' not in i and len(i) > 5)

def get_current_application(self):

"""

获取当前运行的应用信息

:return:

"""

return self.shell('dumpsys window w | %s \/ | %s name=' % (self.__find, self.__find)).read()

def get_current_package(self):

"""

获取当前运行app包名

:return:

"""

reg = re.compile(r'name=(.+?)/')

return re.findall(reg, self.get_current_application())[0]

def get_current_activity(self):

"""

获取当前运行activity

:return: package/activity

"""

reg = re.compile(r'name=(.+?)\)')

return re.findall(reg, self.get_current_application())[0]

def __get_process(self, package_name):

"""

获取进程信息

:param package_name:

:return:

"""

if self.__system is "Windows":

pid_command = self.shell("ps | %s %s$" % (self.__find, package_name)).read()

else:

pid_command = self.shell("ps | %s -w %s" % (self.__find, package_name)).read()

return pid_command

def process_exists(self, package_name):

"""

返回进程是否存在

:param package_name:

:return:

"""

process = self.__get_process(package_name)

return package_name in process

def get_pid(self, package_name):

"""

获取pid

:return:

"""

pid_command = self.__get_process(package_name)

if pid_command == '':

print("The process doesn't exist.")

return pid_command

req = re.compile(r"\d+")

result = str(pid_command).split()

result.remove(result[0])

return req.findall(" ".join(result))[0]

def get_uid(self, pid):

"""

获取uid

:param pid:

:return:

"""

result = self.shell("cat /proc/%s/status" % pid).readlines()

for i in result:

if 'uid' in i.lower():

return i.split()[1]

def get_flow_data_tcp(self, uid):

"""

获取应用tcp流量

:return:(接收, 发送)

"""

tcp_rcv = self.shell("cat proc/uid_stat/%s/tcp_rcv" % uid).read().split()[0]

tcp_snd = self.shell("cat proc/uid_stat/%s/tcp_snd" % uid).read().split()[0]

return tcp_rcv, tcp_snd

def get_flow_data_all(self, uid):

"""

获取应用流量全部数据

包含该应用多个进程的所有数据 tcp udp等

(rx_bytes, tx_bytes) >> (接收, 发送)

:param uid:

:return:list(dict)

"""

all_data = []

d = {}

data = self.shell("cat /proc/net/xt_qtaguid/stats | %s %s" % (self.__find, uid)).readlines()

for i in data:

if not i.startswith('\n'):

item = i.strip().split()

d['idx'] = item[0]

d['iface'] = item[1]

d['acct_tag_hex'] = item[2]

d['uid_tag_int'] = item[3]

d['cnt_set'] = item[4]

d['rx_bytes'] = item[5]

d['rx_packets'] = item[6]

d['tx_bytes'] = item[7]

d['tx_packets'] = item[8]

d['rx_tcp_bytes'] = item[9]

d['rx_tcp_packets'] = item[10]

d['rx_udp_bytes'] = item[11]

d['rx_udp_packets'] = item[12]

d['rx_other_bytes'] = item[13]

d['rx_other_packets'] = item[14]

d['tx_tcp_bytes'] = item[15]

d['tx_tcp_packets'] = item[16]

d['tx_udp_bytes'] = item[17]

d['tx_udp_packets'] = item[18]

d['tx_other_bytes'] = item[19]

d['tx_other_packets'] = item[20]

all_data.append(d)

d = {}

return all_data

@staticmethod

def dump_apk(path):

"""

dump apk文件

:param path: apk路径

:return:

"""

# 检查build-tools是否添加到环境变量中

# 需要用到里面的aapt命令

l = os.environ['PATH'].split(';')

build_tools = False

for i in l:

if 'build-tools' in i:

build_tools = True

if not build_tools:

raise EnvironmentError("ANDROID_HOME BUILD-TOOLS COMMAND NOT FOUND.\nPlease set the environment variable.")

return os.popen('aapt dump badging %s' % (path,))

@staticmethod

def dump_xml(path, filename):

"""

dump apk xml文件

:return:

"""

return os.popen('aapt dump xmlstrings %s %s' % (path, filename))

def uiautomator_dump(self):

"""

获取屏幕uiautomator xml文件

:return:

"""

return self.shell('uiautomator dump').read().split()[-1]

def pull(self, source, target):

"""

从手机端拉取文件到电脑端

:return:

"""

self.adb('pull %s %s' % (source, target))

def push(self, source, target):

"""

从电脑端推送文件到手机端

:param source:

:param target:

:return:

"""

self.adb('push %s %s' % (source, target))

def remove(self, path):

"""

从手机端删除文件

:return:

"""

self.shell('rm %s' % (path,))

def clear_app_data(self, package):

"""

清理应用数据

:return:

"""

self.shell('pm clear %s' % (package,))

def install(self, path):

"""

安装apk文件

:return:

"""

# adb install 安装错误常见列表

errors = {'INSTALL_FAILED_ALREADY_EXISTS': '程序已经存在',

'INSTALL_DEVICES_NOT_FOUND': '找不到设备',

'INSTALL_FAILED_DEVICE_OFFLINE': '设备离线',

'INSTALL_FAILED_INVALID_APK': '无效的APK',

'INSTALL_FAILED_INVALID_URI': '无效的链接',

'INSTALL_FAILED_INSUFFICIENT_STORAGE': '没有足够的存储空间',

'INSTALL_FAILED_DUPLICATE_PACKAGE': '已存在同名程序',

'INSTALL_FAILED_NO_SHARED_USER': '要求的共享用户不存在',

'INSTALL_FAILED_UPDATE_INCOMPATIBLE': '版本不能共存',

'INSTALL_FAILED_SHARED_USER_INCOMPATIBLE': '需求的共享用户签名错误',

'INSTALL_FAILED_MISSING_SHARED_LIBRARY': '需求的共享库已丢失',

'INSTALL_FAILED_REPLACE_COULDNT_DELETE': '需求的共享库无效',

'INSTALL_FAILED_DEXOPT': 'dex优化验证失败',

'INSTALL_FAILED_DEVICE_NOSPACE': '手机存储空间不足导致apk拷贝失败',

'INSTALL_FAILED_DEVICE_COPY_FAILED': '文件拷贝失败',

'INSTALL_FAILED_OLDER_SDK': '系统版本过旧',

'INSTALL_FAILED_CONFLICTING_PROVIDER': '存在同名的内容提供者',

'INSTALL_FAILED_NEWER_SDK': '系统版本过新',

'INSTALL_FAILED_TEST_ONLY': '调用者不被允许测试的测试程序',

'INSTALL_FAILED_CPU_ABI_INCOMPATIBLE': '包含的本机代码不兼容',

'CPU_ABIINSTALL_FAILED_MISSING_FEATURE': '使用了一个无效的特性',

'INSTALL_FAILED_CONTAINER_ERROR': 'SD卡访问失败',

'INSTALL_FAILED_INVALID_INSTALL_LOCATION': '无效的安装路径',

'INSTALL_FAILED_MEDIA_UNAVAILABLE': 'SD卡不存在',

'INSTALL_FAILED_INTERNAL_ERROR': '系统问题导致安装失败',

'INSTALL_PARSE_FAILED_NO_CERTIFICATES': '文件未通过认证 >> 设置开启未知来源',

'INSTALL_PARSE_FAILED_INCONSISTENT_CERTIFICATES': '文件认证不一致 >> 先卸载原来的再安装',

'INSTALL_FAILED_INVALID_ZIP_FILE': '非法的zip文件 >> 先卸载原来的再安装',

'INSTALL_CANCELED_BY_USER': '需要用户确认才可进行安装',

'INSTALL_FAILED_VERIFICATION_FAILURE': '验证失败 >> 尝试重启手机',

'DEFAULT': '未知错误'

}

print('Installing...')

l = self.adb('install -r %s' % (path,)).read()

if 'Success' in l:

print('Install Success')

if 'Failure' in l:

reg = re.compile('\\[(.+?)\\]')

key = re.findall(reg, l)[0]

try:

print('Install Failure >> %s' % errors[key])

except KeyError:

print('Install Failure >> %s' % key)

return l

def uninstall(self, package):

"""

卸载apk

:param package: 包名

:return:

"""

print('Uninstalling...')

l = self.adb('uninstall %s' % (package,)).read()

print(l)

def screenshot(self, target_path=''):

"""

手机截图

:param target_path: 目标路径

:return:

"""

format_time = utils.timetools.timestamp('%Y%m%d%H%M%S')

self.shell('screencap -p /sdcard/%s.png' % (format_time,))

time.sleep(1)

if target_path == '':

self.pull('/sdcard/%s.png' % (format_time,), os.path.expanduser('~'))

else:

self.pull('/sdcard/%s.png' % (format_time,), target_path)

self.remove('/sdcard/%s.png' % (format_time,))

def get_cache_logcat(self):

"""

导出缓存日志

:return:

"""

return self.adb('logcat -v time -d')

def get_crash_logcat(self):

"""

导出崩溃日志

:return:

"""

return self.adb('logcat -v time -d | %s AndroidRuntime' % (self.__find,))

def clear_cache_logcat(self):

"""

清理缓存区日志

:return:

"""

self.adb('logcat -c')

def get_device_time(self):

"""

获取设备时间

:return:

"""

return self.shell('date').read().strip()

def ls(self, command):

"""

shell ls命令

:return:

"""

return self.shell('ls %s' % (command,)).readlines()

def file_exists(self, target):

"""

判断文件在目标路径是否存在

:return:

"""

l = self.ls(target)

for i in l:

if i.strip() == target:

return True

return False

def is_install(self, target_app):

"""

判断目标app在设备上是否已安装

:param target_app: 目标app包名

:return: bool

"""

return target_app in self.shell('pm list packages %s' % (target_app,)).read()

def get_device_model(self):

"""

获取设备型号

:return:

"""

return self.shell('getprop ro.product.model').read().strip()

def get_device_id(self):

"""

获取设备id

:return:

"""

return self.adb('get-serialno').read().strip()

def get_device_android_version(self):

"""

获取设备Android版本

:return:

"""

return self.shell('getprop ro.build.version.release').read().strip()

def get_device_sdk_version(self):

"""

获取设备SDK版本

:return:

"""

return self.shell('getprop ro.build.version.sdk').read().strip()

def get_device_mac_address(self):

"""

获取设备MAC地址

:return:

"""

return self.shell('cat /sys/class/net/wlan0/address').read().strip()

def get_device_ip_address(self):

"""

获取设备IP地址

pass: 适用WIFI 蜂窝数据

:return:

"""

if not self.get_wifi_state() and not self.get_data_state():

return

l = self.shell('ip addr | %s global' % self.__find).read()

reg = re.compile('\d+\.\d+\.\d+\.\d+')

return re.findall(reg, l)[0]

def get_device_imei(self):

"""

获取设备IMEI

:return:

"""

sdk = self.get_device_sdk_version()

# Android 5.0以下方法

if int(sdk) < 21:

l = self.shell('dumpsys iphonesubinfo').read()

reg = re.compile('[0-9]{15}')

return re.findall(reg, l)[0]

elif self.root():

l = self.shell('service call iphonesubinfo 1').read()

print(l)

print(re.findall(re.compile("'.+?'"), l))

imei = ''

for i in re.findall(re.compile("'.+?'"), l):

imei += i.replace('.', '').replace("'", '').replace(' ', '')

return imei

else:

print('The device not root.')

return ''

def check_sim_card(self):

"""

检查设备SIM卡

:return:

"""

return len(self.shell('getprop | %s gsm.operator.alpha]' % self.__find).read().strip().split()[-1]) > 2

def get_device_operators(self):

"""

获取运营商

:return:

"""

return self.shell('getprop | %s gsm.operator.alpha]' % self.__find).read().strip().split()[-1]

def get_device_state(self):

"""

获取设备状态

:return:

"""

return self.adb('get-state').read().strip()

def get_display_state(self):

"""

获取屏幕状态

:return: 亮屏/灭屏

"""

l = self.shell('dumpsys power').readlines()

for i in l:

if 'mScreenOn=' in i:

return i.split()[-1] == 'mScreenOn=true'

if 'Display Power' in i:

return 'ON' in i.split('=')[-1].upper()

def get_screen_normal_size(self):

"""

获取设备屏幕分辨率 >> 标配

:return:

"""

return self.shell('wm size').read().strip().split()[-1].split('x')

def get_screen_reality_size(self):

"""

获取设备屏幕分辨率 >> 实际分辨率

:return:

"""

x = 0

y = 0

l = self.shell(r'getevent -p | %s -e "0"' % self.__find).readlines()

for n in l:

if len(n.split()) > 0:

if n.split()[0] == '0035':

x = int(n.split()[7].split(',')[0])

elif n.split()[0] == '0036':

y = int(n.split()[7].split(',')[0])

return x, y

def get_device_interior_sdcard(self):

"""

获取内部SD卡空间

:return: (path,total,used,free,block)

"""

return self.shell('df | %s \/mnt\/shell\/emulated' % self.__find).read().strip().split()

def get_device_external_sdcard(self):

"""

获取外部SD卡空间

:return: (path,total,used,free,block)

"""

return self.shell('df | %s \/storage' % self.__find).read().strip().split()

def __fill_rom(self, path, stream, count):

"""

填充数据

:param path: 填充地址

:param stream: 填充流大小

:param count: 填充次数

:return:

"""

self.shell('dd if=/dev/zero of=%s bs=%s count=%s' % (path, stream, count)).read().strip()

def fill_interior_sdcard(self, filename, size):

"""

填充内置SD卡

:param filename: 文件名

:param size: 填充大小,单位byte

:return:

"""

if size > 10485760: # 10m

self.__fill_rom('sdcard/%s' % filename, 10485760, size / 10485760)

else:

self.__fill_rom('sdcard/%s' % filename, size, 1)

def fill_external_sdcard(self, filename, size):

"""

填充外置SD卡

:param filename: 文件名

:param size: 填充大小,单位byte

:return:

"""

path = self.get_device_external_sdcard()[0]

if size > 10485760: # 10m

self.__fill_rom('%s/%s' % (path, filename), 10485760, size / 10485760)

else:

self.__fill_rom('%s/%s' % (path, filename), size, 1)

def kill_process(self, pid):

"""

杀死进程

pass: 一般需要权限不推荐使用

:return:

"""

return self.shell('kill %s' % pid).read().strip()

def quit_app(self, package):

"""

退出应用

:return:

"""

return self.shell('am force-stop %s' % package).read().strip()

def reboot(self):

"""

重启设备

:return:

"""

self.adb('reboot')

def recovery(self):

"""

重启设备并进入recovery模式

:return:

"""

self.adb('reboot recovery')

def fastboot(self):

"""

重启设备并进入fastboot模式

:return:

"""

self.adb('reboot bootloader')

def root(self):

"""

获取root状态

:return:

"""

return 'not found' not in self.shell('su -c ls -l /data/').read().strip()

def wifi(self, power):

"""

开启/关闭wifi

pass: 需要root权限

:return:

"""

if not self.root():

print('The device not root.')

return

if power:

self.shell('su -c svc wifi enable').read().strip()

else:

self.shell('su -c svc wifi disable').read().strip()

def data(self, power):

"""

开启/关闭蜂窝数据

pass: 需要root权限

:return:

"""

if not self.root():

print('The device not root.')

return

if power:

self.shell('su -c svc data enable').read().strip()

else:

self.shell('su -c svc data disable').read().strip()

def get_wifi_state(self):

"""

获取WiFi连接状态

:return:

"""

return 'enabled' in self.shell('dumpsys wifi | %s ^Wi-Fi' % self.__find).read().strip()

def get_data_state(self):

"""

获取移动网络连接状态

:return:

"""

return '2' in self.shell('dumpsys telephony.registry | %s mDataConnectionState' % self.__find).read().strip()

def get_network_state(self):

"""

设备是否连上互联网

:return:

"""

return 'unknown host' not in self.shell('ping -w 1 www.baidu.com').read().strip()

def get_wifi_password_list(self):

"""

获取WIFI密码列表

:return:

"""

if not self.root():

print('The device not root.')

return []

l = re.findall(re.compile('ssid=".+?"\s{3}psk=".+?"'), self.shell('su -c cat /data/misc/wifi/*.conf').read())

return [re.findall(re.compile('".+?"'), i) for i in l]

def call(self, number):

"""

拨打电话

:param number:

:return:

"""

self.shell('am start -a android.intent.action.CALL -d tel:%s' % number)

def open_url(self, url):

"""

打开网页

:return:

"""

self.shell('am start -a android.intent.action.VIEW -d %s' % url)

def start_application(self, component):

"""

启动一个应用

e.g: com.android.settings/com.android.settings.Settings

"""

self.shell("am start -n %s" % component)

def send_keyevent(self, keycode):

"""

发送一个按键事件

https://developer.android.com/reference/android/view/KeyEvent.html

:return:

"""

self.shell('input keyevent %s' % keycode)

def rotation_screen(self, param):

"""

旋转屏幕

:param param: 0 >> 纵向,禁止自动旋转; 1 >> 自动旋转

:return:

"""

self.shell('/system/bin/content insert --uri content://settings/system --bind '

'name:s:accelerometer_rotation --bind value:i:%s' % param)

def instrument(self, command):

"""

启动instrument app

:param command: 命令

:return:

"""

return self.shell('am instrument %s' % command).read()

def export_apk(self, package, target_path='', timeout=5000):

"""

从设备导出应用

:param timeout: 超时时间

:param target_path: 导出后apk存储路径

:param package: 包名

:return:

"""

num = 0

if target_path == '':

self.adb('pull /data/app/%s-1/base.apk %s' % (package, os.path.expanduser('~')))

while 1:

num += 1

if num <= timeout:

if os.path.exists(os.path.join(os.path.expanduser('~'), 'base.apk')):

os.rename(os.path.join(os.path.expanduser('~'), 'base.apk'),

os.path.join(os.path.expanduser('~'), '%s.apk' % package))

else:

self.adb('pull /data/app/%s-1/base.apk %s' % (package, target_path))

while 1:

num += 1

if num <= timeout:

if os.path.exists(os.path.join(os.path.expanduser('~'), 'base.apk')):

os.rename(os.path.join(os.path.expanduser('~'), 'base.apk'),

os.path.join(os.path.expanduser('~'), '%s.apk' % package))

class KeyCode:

KEYCODE_CALL = 5 # 拨号键

KEYCODE_ENDCALL = 6 # 挂机键

KEYCODE_HOME = 3 # Home键

KEYCODE_MENU = 82 # 菜单键

KEYCODE_BACK = 4 # 返回键

KEYCODE_SEARCH = 84 # 搜索键

KEYCODE_CAMERA = 27 # 拍照键

KEYCODE_FOCUS = 80 # 对焦键

KEYCODE_POWER = 26 # 电源键

KEYCODE_NOTIFICATION = 83 # 通知键

KEYCODE_MUTE = 91 # 话筒静音键

KEYCODE_VOLUME_MUTE = 164 # 扬声器静音键

KEYCODE_VOLUME_UP = 24 # 音量+键

KEYCODE_VOLUME_DOWN = 25 # 音量-键

KEYCODE_ENTER = 66 # 回车键

KEYCODE_ESCAPE = 111 # ESC键

KEYCODE_DPAD_CENTER = 23 # 导航键 >> 确定键

KEYCODE_DPAD_UP = 19 # 导航键 >> 向上

KEYCODE_DPAD_DOWN = 20 # 导航键 >> 向下

KEYCODE_DPAD_LEFT = 21 # 导航键 >> 向左

KEYCODE_DPAD_RIGHT = 22 # 导航键 >> 向右

KEYCODE_MOVE_HOME = 122 # 光标移动到开始键

KEYCODE_MOVE_END = 123 # 光标移动到末尾键

KEYCODE_PAGE_UP = 92 # 向上翻页键

KEYCODE_PAGE_DOWN = 93 # 向下翻页键

KEYCODE_DEL = 67 # 退格键

KEYCODE_FORWARD_DEL = 112 # 删除键

KEYCODE_INSERT = 124 # 插入键

KEYCODE_TAB = 61 # Tab键

KEYCODE_NUM_LOCK = 143 # 小键盘锁

KEYCODE_CAPS_LOCK = 115 # 大写锁定键

KEYCODE_BREAK = 121 # Break / Pause键

KEYCODE_SCROLL_LOCK = 116 # 滚动锁定键

KEYCODE_ZOOM_IN = 168 # 放大键

KEYCODE_ZOOM_OUT = 169 # 缩小键

KEYCODE_0 = 7

KEYCODE_1 = 8

KEYCODE_2 = 9

KEYCODE_3 = 10

KEYCODE_4 = 11

KEYCODE_5 = 12

KEYCODE_6 = 13

KEYCODE_7 = 14

KEYCODE_8 = 15

KEYCODE_9 = 16

KEYCODE_A = 29

KEYCODE_B = 30

KEYCODE_C = 31

KEYCODE_D = 32

KEYCODE_E = 33

KEYCODE_F = 34

KEYCODE_G = 35

KEYCODE_H = 36

KEYCODE_I = 37

KEYCODE_J = 38

KEYCODE_K = 39

KEYCODE_L = 40

KEYCODE_M = 41

KEYCODE_N = 42

KEYCODE_O = 43

KEYCODE_P = 44

KEYCODE_Q = 45

KEYCODE_R = 46

KEYCODE_S = 47

KEYCODE_T = 48

KEYCODE_U = 49

KEYCODE_V = 50

KEYCODE_W = 51

KEYCODE_X = 52

KEYCODE_Y = 53

KEYCODE_Z = 54

KEYCODE_PLUS = 81 # +

KEYCODE_MINUS = 69 # -

KEYCODE_STAR = 17 # *

KEYCODE_SLASH = 76 # /

KEYCODE_EQUALS = 70 # =

KEYCODE_AT = 77 # @

KEYCODE_POUND = 18 # #

KEYCODE_APOSTROPHE = 75 # '

KEYCODE_BACKSLASH = 73 # \

KEYCODE_COMMA = 55 # ,

KEYCODE_PERIOD = 56 # .

KEYCODE_LEFT_BRACKET = 71 # [

KEYCODE_RIGHT_BRACKET = 72 # ]

KEYCODE_SEMICOLON = 74 # ;

KEYCODE_GRAVE = 68 # `

KEYCODE_SPACE = 62 # 空格键

KEYCODE_MEDIA_PLAY = 126 # 多媒体键 >> 播放

KEYCODE_MEDIA_STOP = 86 # 多媒体键 >> 停止

KEYCODE_MEDIA_PAUSE = 127 # 多媒体键 >> 暂停

KEYCODE_MEDIA_PLAY_PAUSE = 85 # 多媒体键 >> 播放 / 暂停

KEYCODE_MEDIA_FAST_FORWARD = 90 # 多媒体键 >> 快进

KEYCODE_MEDIA_REWIND = 89 # 多媒体键 >> 快退

KEYCODE_MEDIA_NEXT = 87 # 多媒体键 >> 下一首

KEYCODE_MEDIA_PREVIOUS = 88 # 多媒体键 >> 上一首

KEYCODE_MEDIA_CLOSE = 128 # 多媒体键 >> 关闭

KEYCODE_MEDIA_EJECT = 129 # 多媒体键 >> 弹出

KEYCODE_MEDIA_RECORD = 130 # 多媒体键 >> 录音

if __name__ == '__main__':

a = AdbTools()

pass

补充知识:Python调用adb命令实现对多台设备同时进行reboot

首先,adb实现对设备的reboot命令是:adb reboot . 但是如果是两台/多台设备的时候,需要声明serial number: adb -s serial_no reboot.

那么,如何用python实现对多台设备进行adb操作呢(reboot)?

这里涉及到 python 下 subprocess model的使用:

import subprocess

adb device 获取所有设备的 serial number:

devices = subprocess.Popen(

'adb devices'.split(),

stdout=subprocess.PIPE,

stderr=subprocess.PIPE

).communicate()[0]

这样adb device命令的返回信息都在devices下,但是我们只需要 serial number的:

serial_nos = []

for item in devices.split():

filters = ['list', 'of', 'device', 'devices', 'attached']

if item.lower() not in filters:

serial_nos.append(item)

这样serial_nos 下保存的就是所有设备的 serial number 了,下面我们只需要依次对其进行adb -s [serial_number] reboot即可:

for serial_no in serial_nos:

reboot_cmds.append('adb -s %s reboot' % serial_no)

for reboot_cmd in reboot_cmds:

subprocess.Popen(

reboot_cmd.split(),

stdout=subprocess.PIPE,

stderr=subprocess.PIPE

).communicate()[0]

这样,每个设备都进行了reboot的操作了……

以上这篇Python实现对adb命令封装就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持脚本之家。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/340672.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ubuntu优化服务器网络连接数,ubuntu下解决高并发socket最大连接数限制,tcp默认1024个连接...

linux系统默认ulimit为1024个访问 用户最多可开启的程序数目。一般一个端口(即一个进程)的最高连接为2的16次方65536通过这个命令 ulimit -n 可以看到默认值为1024查看全局文件句柄数限制(系统支持的最大值)cat /proc/sys/fs/file-max查看每个进程文件句柄数限制ulimit -n第一步…

c++中求解非线性方程组_齐次线性方程组的基础解系的简便算法

线性方程组的求解是线性代数中的基本技能&#xff0c;而齐次线性方程组的基础解系的求法又是基础。本文给出一个计算齐次线性方程组的基础解系的公式&#xff0c;从而简化计算过程。01 符号说明 n元线性方程组的矩阵形式&#xff1a;(1)齐次线性方程组;(2)非齐次线性方程组;系数…

在AWS第1部分中使用Terraform自动缩放组:基本步骤

因此&#xff0c;您想使用Terraform在AWS上创建一个自动缩放组。 以下是达到此目的的最少步骤。 在编写实际代码之前&#xff0c;应指定aws terraform提供程序以及provider.tf文件上的区域。 provider "aws" { version "~> 2.0" region "eu-wes…

python的if语句例句_Python入门之if条件语句

Besides the while statement just introduced, Python knows the usual control flow statements known from other languages, with some twists.除了之前介绍的while语句&#xff0c;Python同样支持其他语言通常用的控制流语句&#xff0c;但也有一些区别。 if Statements P…

虚拟空间独立服务器哪个好,共享虚拟主机和独立哪个好

共享虚拟主机和独立哪个好&#xff1f;现在市面上的虚拟主机产品分为共享主机和独立虚拟主机&#xff0c;那对于用户来说&#xff0c;他们二者之间哪个更好些呢。共享虚拟主机共享虚拟主机指多个用户共同平均使用同一台服务器资源&#xff0c;包括CPU、内存、带宽、硬盘和IP都是…

esp32 怎么分配freertos 堆栈大小_深度解剖~ FreeRtos阅读笔记2 任务创建、内核链表初始化...

2.FREERTOS任务创建、内核链表初始化硬件环境&#xff1a;cortex m4FreeRTOS版本:v8.0.1今天开始阅读freertos&#xff0c;阅读同时做下笔记&#xff0c;等哪天碰到移植问题再翻出来看看。2.1 任务、链表结构体源码中使用tskTCB来存储一个任务的所有信息&#xff0c;xLIST存储内…

pandas用众数填充缺失值_7步搞定数据清洗-Python数据清洗指南

脏数据就是在物理上临时存在过&#xff0c;但在逻辑上不存在的数据。数据清洗是整个数据分析过程的第一步&#xff0c;就像做一道菜之前需要先择菜洗菜一样。数据分析师经常需要花费大量的时间来清洗数据或者转换格式&#xff0c;这个工作甚至会占整个数据分析流程的80%左右的时…

有盘和无盘服务器,网吧系统的终极解决之道:无盘系统+有盘(转)

1、效果好&#xff1a;系统启动飞速&#xff0c;玩大型游戏飞速&#xff0c;不会丢失账号&#xff0c;系统永远干净。2、不会中毒&#xff1a;系统文件从服务器上的镜像中读取&#xff0c;大型游戏从本地硬盘读取&#xff0c;本地硬盘全盘还原&#xff0c;游戏穿透更新。3、投资…

Sigma IDE现在支持Python无服务器Lambda函数!

想想无服务器&#xff0c;使用Pythonic –全部在您的浏览器中&#xff01; &#xff08;好吧&#xff0c;这则新闻已经过了几周了&#xff0c;但是仍然……&#xff09; 如果您沉迷于整个无服务器的“事物”中 &#xff0c;您可能已经注意到我们&#xff0c;一个在SLAppForge臭…

微信分享 ajax冲突,微信jssdk分享功能开发及解决ajax跨域的问题

微信JS-SDK说明文档微信JS-SDK Demo我所要实现的分享功能要求&#xff1a;只要提供一段js代码给前端美工放置在静态页面等就可以实现分享功能。js代码如下&#xff1a;urllocation.href;$.ajax({type : "get",url : "http://域名/wx/test/jssdk.php?url"u…

idle不是python自带的开发工具_Python的开发工具

通常情况下&#xff0c;为了提高开发效率&#xff0c;需要使用相应的开发工具。进行Python开发也可以使用开发工具。下面将详细介绍Python自带的IDLE 一使用自带的IDLE 在安装Python后&#xff0c;会自动安装一个IDLE。它是一个Python shell(可以在打开的IDLE窗口的标题栏上看到…

合肥工业大学机器人技术期末_机器人技术第三次作业(HFUT)

第三次作业本人代码水平十分有限&#xff0c;仅供参考&#xff0c;有错误请指出java源码&#xff1a;package robathomework3;import java.lang.Math;//点类class point {double x 0;double y 0;point(double x, double y) {this.x x;this.y y;}void print() {System.out.p…

java se 导原码_Java SE 8新功能导览:Java开发世界中的重大变化

java se 导原码我很自豪&#xff0c;像其他专业团队成员一样&#xff0c;是采用OpenJDK的成员之一&#xff0c;但是从过去8个月就加入了&#xff0c;我们经历了Java SE 8 开发&#xff0c;编译&#xff0c;编码&#xff0c;讨论等各个阶段&#xff0c;直到将其付诸实践为止。 。…

linux将日期和日历信息追加到文件中_Linux任务调度

crontab 任务调度crontab 进行定时任务的设置概述 任务调度&#xff1a;是指系统在某个时间执行的特定的命令或程序。任务调度分类&#xff1a;1.系统工作&#xff1a;有些重要的工作必须周而复始地执行&#xff0c;如病毒扫描等 。2.个别用户工作&#xff1a;个别用户可能希望…

android sdk build-tools_从零开始仿写一个抖音App——视频编辑SDK开发(一)

本文首发于微信公众号——世界上有意思的事&#xff0c;搬运转载请注明出处&#xff0c;否则将追究版权责任。交流qq群&#xff1a;859640274。大家好久不见&#xff0c;又有一个多月没有发文章了。不知道还有哪些读者记得我的 从零开始仿写抖音App 的系列文章&#xff0c;这个…

爱默生E系列服务器机柜托盘,艾默生通信电源PS48300-3B/1800 一体化室内机柜

PS48300-3B/1800电源系统PS48300-3B/1800电源系统是艾默生网络能源集多年开发和网上运行经验&#xff0c;采用 DSP控制技术&#xff0c;为满足3G网络需求而设计的高可靠、高功率密度、高性能、全数 字化通信电源系统。根据交流配电和机柜高度。一、特点 1、休眠节能专利技术&am…

功能Java示例 第8部分–更多纯函数

这是第8部分&#xff0c;该系列的最后一部分称为“ Functional Java by Example”。 我在本系列的每个部分中发展的示例是某种“提要处理程序”&#xff0c;用于处理文档。 在上一期文章中&#xff0c;我们已经使用Vavr库看到了一些模式匹配&#xff0c;并且还将故障也视为数据…

tensorflow 小于_坐姿不对,屏幕就变模糊!教你用TensorFlow做一款“隐形背背佳”...

大数据文摘出品作为一个上班族&#xff0c;每天坐在电脑前那么久&#xff0c;难免出现腰酸背痛的情况&#xff0c;时间长了甚至脊柱都歪曲变形了&#xff0c;这可不行&#xff01;一定要克制住自己的坐姿。这里有款“隐形背背佳”&#xff0c;要不要了解一下&#xff1f;一位名…

python处理csv文件列错位_CSV文件分割与列异常处理的python脚本

#codingutf-8 ‘‘‘脚本说明 用来解决csv文件的列异常问题&#xff08;以逗号分隔符为例&#xff09;&#xff1a; csv文件有些列含有换行符、逗号等特殊符号&#xff0c;这就导致csv文件出现列异常的情况。 此脚本将csv文件输出成如下文件&#xff1a;源文件名-正确列-文件序…

文本分类和提取关键词算法_文本内容之间的关键词提取和相似度计算

文本分类和提取关键词算法背景 Web应用程序变得越来越智能。 从网站上使用服务的日子已经一去不复返了&#xff0c;用户不得不填写一个巨大的表格。 假设您有一个适合书迷的网站。 在Web 2.0之前&#xff0c;像这样的网站曾经以诸如年龄&#xff0c;阅读的书籍&#xff0c;喜欢…