国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術文章
文章詳情頁

Python實現對adb命令封裝

瀏覽:4日期:2022-08-03 13:47:50

我就廢話不多說了,大家還是直接看代碼吧!

#!/usr/bin/evn python# -*- coding:utf-8 -*- # FileName adbtools.py# Author: HeyNiu# Created Time: 2016/9/19'''adb 工具類''' import osimport platformimport reimport timeimport 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 = pathelse: 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 = pathelse: 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 += 1if 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 += 1if 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命令封裝就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持好吧啦網。

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 97视频在线免费播放 | 亚洲美女影院 | 一级a俄罗斯毛片免费 | 亚洲三级成人 | 午夜大片免费男女爽爽影院久久 | 亚洲精彩 | 欧洲免费无线码二区5 | 国产成人亚洲毛片 | 国产日b视频 | 八戒午夜精品视频在线观看 | 美女动作一级毛片 | 国产亚洲欧美成人久久片 | 国内精品免费一区二区观看 | 日本一级特黄特色大片免费视频 | 国产精品爱久久久久久久三级 | 亚洲成年男人的天堂网 | 久久狠狠一本精品综合网 | 经典三级久久 | 国产成人最新毛片基地 | 国产三级在线观看 | 福利一二三区 | 99热久久国产这里是精品 | 国产精品19禁在线观看2021 | 国产综合久久久久影院 | www亚洲免费 | 91亚洲精品在看在线观看高清 | 欧美一级片在线免费观看 | 成人国产在线观看 | 亚洲精品98久久久久久中文字幕 | 欧美白人最猛性xxxxx | 亚洲天堂一区二区在线观看 | 人与禽的免费一级毛片 | 日本一区二区高清不卡 | 黄黄的网站在线观看 | 一级片视频免费看 | 久久视频精品线视频在线网站 | 成人女人a毛片在线看 | 国产在线观看免费人成小说 | 久久久久久国产精品免费免 | 亚洲韩国日本欧美一区二区三区 | 国产大臿蕉香蕉大视频 |