每年淘宝双十一的时候,总是要刷各种各样的浏览页面,收集能量或者喵币或者什么。
那既然如此,我就总想着,能否通过Python自动调用的方式来刷网页。
本文是基于使用Python控制手机(一),默认已经安装了ADB并配置了环境变量,安装了Python环境,且在Python中安装了uiautomator2和weditor等包。
当我们使用uiautomator2包来打开某个APP时,可以通过点击屏幕特殊位置的方式来实现。但是其中存在的问题便是,可能由于我们APP图标的移动,而使得程序无法运行。健壮性和通用性不高。
其实在uiautomator2这个包中,提供了一种可以通过APP包名就可以打开特定APP的方式,例如打开和关闭淘宝。
import uiautomator2 as u2
import time
d = u2.connect() # 连接设备
d.app_start("com.taobao.taobao") # 打开淘宝
time.sleep(10) # 等待10秒钟
d.app_stop("com.taobao.taobao") # 关闭淘宝
再比如打开和关闭微信:
import uiautomator2 as u2
import time
d = u2.connect() # 连接设备
d.app_start("com.tencent.mm") # 打开微信
time.sleep(10) # 等待10秒钟
d.app_stop("com.tencent.mm") # 关闭微信
有的时候,我们是不太清楚一个APP的包名的,这时我们可以通过打印设备当前信息的方式来获取APP的包名。首先我们需要将要获取的APP打开,并且保持在手机最前台,执行代码:
import uiautomator2 as u2
import time
d = u2.connect() # 连接设备
print(d.info) # 打印设备信息
输出结果如下:
{'currentPackageName': 'com.taobao.taobao', 'displayHeight': 2111, 'displayRotation': 0, 'displaySizeDpX': 393, 'displaySizeDpY': 851, 'displayWidth': 1080, 'productName': 'cannon', 'screenOn': True, 'sdkInt': 29, 'naturalOrientation': True}
Process finished with exit code 0
在所打印的Json键值对中,键currentPackageName对应的值,即为此时正在最前台的APP的包名,上述结果操作时,正在最前的APP为淘宝。
一般来说,如果页面切换按钮含有特定文字,我们直接通过文字进行定位是最方便的,也是最准确的,比如打开微信朋友圈:
import uiautomator2 as u2
import time
d = u2.connect() # 连接设备
d.app_start("com.tencent.mm") # 打开微信
time.sleep(2) # 等待2秒钟
d(text='发现').click() # 点击文字为“发现”的控件
time.sleep(2) # 等待2秒钟
d(text='朋友圈').click() # 点击文字为“朋友圈”的控件
因为可能存在的,APP的加载时间和对点击操作的响应时间,尽量在每次点击操作之后,为APP和手机留有足够的反应时间。值得注意的是,如果打开微信之后,恰好有个常用联系人的昵称叫做“发现”,那就可能会被误点,这种情况下我们需要使用别的定位方式来定位特定控件。
需要点击的文字如果是固定的,就可以使用d(text="XXX")来选择控件元素,其中XXX为特定的文字。如果部分文字是固定的,比如第一次元素显示文字为“我是第11932位访客”,第二次显示文字为“我是第12111位访客”,那我们可以通过d(textContains="我是第").click()来点击这个控件,或者通过d(textContains="位访客").click()来点击这个控件,这种方式就可以通过子字符串来定位特定的元素控件。
还是使用进入朋友圈举例:
import uiautomator2 as u2
import time
d = u2.connect() # 连接设备
d.app_start("com.tencent.mm") # 打开微信
time.sleep(2) # 等待2秒钟
# 点击“发现”,三选一
d(text='发现').click() # 点击文字为“发现”控件
d(textContains='发').click() # 点击带“发”的控件
# 通过WEditor获得的xpath定位
d.xpath('//*[@resource-id="com.tencent.mm:id/e8y"]/android.widget.LinearLayout[1]/android.widget.RelativeLayout[3]/android.widget.LinearLayout[1]').click()
time.sleep(2) # 等待2秒钟
# 点击“朋友圈”,三选一
d(text='朋友圈').click() # 点击文字为“朋友圈”控件
d(textContains='朋').click() # 点击带“朋”的控件
# 通过WEditor获得的xpath定位
d.xpath('//*[@resource-id="android:id/list"]/android.widget.LinearLayout[1]/android.widget.LinearLayout[1]/android.widget.LinearLayout[1]/android.widget.LinearLayout[1]/android.widget.LinearLayout[1]/android.widget.LinearLayout[1]/android.widget.LinearLayout[1]').click()
其实还有很多各不相同的定位方式,只要能定位到唯一的特定的控件进行点击即可。例如在双十一时,我在淘宝中切换到收集喵币页面的点击事件:
import uiautomator2 as u2
import time
d = u2.connect() # 连接设备
d.app_start("com.taobao.taobao") # 打开淘宝
time.sleep(5) # 等待5秒钟
d.xpath('//*[@content-desc="双11超级喵糖"]').click() # 打开喵糖页面
在双十一淘宝活动中,打开喵糖页面,会先弹出提示是否将此页面加入收藏的弹框,点击文字为“我再想想”的按钮。注意要判断此控件是否存在,点击不存在的控件将会报错。如果不确定一个控件是否存在,又不想判断的情况下,则需要使用 try......catch...... 来将其包裹。
if len(d(textContains='我再想想')) > 0: # 如果存在此控件
d(textContains='我再想想').click() # 点击“我再想想”
点击“赚糖”控件,因为这个控件经常会被屏幕上出现的手指动画所挡住,因此需要等待:
while len(d(textContains='赚糖')) <= 0:
time.sleep(1)
d(textContains='赚糖').click()
然后点击完后等会儿,再点击“去浏览”按钮:
while len(d(textContains='去浏览')) > 0:
print("检测到浏览按钮...")
d(textContains='去浏览').click()
等待15秒(算上反应时间,需要多等一会儿)返回即可:
d.press("back") # 相当于手机返回键
其实具体的部分实现起来比较简单,在此总结一下uiautomator2 的其它一些功能。
关于按键:
d.press("home") # 点击home键
d.press("back") # 点击back键
d.press("left") # 点击左键
d.press("right") # 点击右键
d.press("up") # 点击上键
d.press("down") # 点击下键
d.press("center") # 点击选中
d.press("menu") # 点击menu按键
d.press("search") # 点击搜索按键
d.press("enter") # 点击enter键
d.press("delete") # 点击删除按键
d.press("recent") # 点击近期活动按键
d.press("volume_up") # 音量+
d.press("volume_down") # 音量-
d.press("volume_mute") # 静音
d.press("camera") # 相机
d.press("power") # 电源键
关于锁屏与解锁:
# 一个设备信息字典中的布尔值,为true时代表当前屏幕亮起,为false代表当前屏幕熄灭
d.info.get('screenOn')
# 仅点亮屏幕
d.screen_on()
# 点亮屏幕并解锁,注意如果有密码,则只能进入密码输入页面,需要输入密码才能解锁
d.unlock()
# 关闭屏幕
d.screen_off()
关于点击等操作(支持百分比):
# 单击屏幕
d.click(x,y) # x,y为点击坐标
# 双击屏幕
d.double_click(x, y)
d.double_click(x, y, 0.1) # 默认两个单击之间间隔时间为0.1秒
# 长按
d.long_click(x, y)
d.long_click(x, y, 0.5) # 长按0.5秒(默认)
# 滑动
d.swipe(sx, sy, ex, ey)
d.swipe(sx, sy, ex, ey, 0.5) # 滑动0.5秒(默认)
#拖动
d.drag(sx, sy, ex, ey)
d.drag(sx, sy, ex, ey, 0.5) # 拖动0.5秒(默认)
# 滑动点 多用于九宫格解锁,提前获取到每个点的相对坐标(这里支持百分比)
# 从点(x0, y0)滑到点(x1, y1)再滑到点(x2, y2)
# 两点之间的滑动速度是0.2秒
d.swipe((x0, y0), (x1, y1), (x2, y2), 0.2)
# 注意:单击,滑动,拖动操作支持百分比位置值。例:
d.long_click(0.5, 0.5) 表示长按屏幕中心
当然还有其它的一些功能,例如向上滑动屏幕,直到指定文字出现为止:
d(scrollable=True).scroll.to(text="3年级2班")
等等。基本上都可以比较容易的百度到,在此仅简单列举一二。