1. 序言
每年淘宝双十一的时候,总是要刷各种各样的浏览页面,收集能量或者喵币或者什么。
那既然如此,我就总想着,能否通过Python自动调用的方式来刷网页。
2. 前置工作
本文是基于使用Python控制手机(一),默认已经安装了ADB并配置了环境变量,安装了Python环境,且在Python中安装了uiautomator2和weditor等包。
3. 打开APP
当我们使用uiautomator2包来打开某个APP时,可以通过点击屏幕特殊位置的方式来实现。但是其中存在的问题便是,可能由于我们APP图标的移动,而使得程序无法运行。健壮性和通用性不高。
其实在uiautomator2这个包中,提供了一种可以通过APP包名就可以打开特定APP的方式,例如打开和关闭淘宝。
1
2
3
4
5
6
|
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" ) # 关闭淘宝
|
再比如打开和关闭微信:
1
2
3
4
5
6
|
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" ) # 关闭微信
|
4. 获取APP的包名
有的时候,我们是不太清楚一个APP的包名的,这时我们可以通过打印设备当前信息的方式来获取APP的包名。首先我们需要将要获取的APP打开,并且保持在手机最前台
执行代码:
1
2
3
4
|
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为淘宝。
5. 打开特定页面
一般来说,如果页面切换按钮含有特定文字,我们直接通过文字进行定位是最方便的,也是最准确的,比如打开微信朋友圈:
1
2
3
4
5
6
7
8
|
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()来点击这个控件,这种方式就可以通过子字符串来定位特定的元素控件。
还是使用进入朋友圈举例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
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()
|
其实还有很多各不相同的定位方式,只要能定位到唯一的特定的控件进行点击即可。例如在双十一时,我在淘宝中切换到收集喵币页面的点击事件:
1
2
3
4
5
6
|
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() # 打开喵糖页面
|
6. 其它细节
在双十一淘宝活动中,打开喵糖页面,会先弹出提示是否将此页面加入收藏的弹框,点击文字为“我再想想”的按钮。注意要判断此控件是否存在,点击不存在的控件将会报错。如果不确定一个控件是否存在,又不想判断的情况下,则需要使用 try......catch...... 来将其包裹。
1
2
|
if len (d(textContains = '我再想想' )) > 0 : # 如果存在此控件
d(textContains = '我再想想' ).click() # 点击“我再想想”
|
点击“赚糖”控件,因为这个控件经常会被屏幕上出现的手指动画所挡住,因此需要等待:
1
2
3
|
while len (d(textContains = '赚糖' )) < = 0 :
time.sleep( 1 )
d(textContains = '赚糖' ).click()
|
然后点击完后等会儿,再点击“去浏览”按钮:
1
2
3
|
while len (d(textContains = '去浏览' )) > 0 :
print ( "检测到浏览按钮..." )
d(textContains = '去浏览' ).click()
|
等待15秒(算上反应时间,需要多等一会儿)返回即可:
1
|
d.press( "back" ) # 相当于手机返回键
|
7. 总结
其实具体的部分实现起来比较简单,在此总结一下uiautomator2 的其它一些功能。
关于按键:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
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" ) # 电源键
|
关于锁屏与解锁:
1
2
3
4
5
6
7
8
|
# 一个设备信息字典中的布尔值,为true时代表当前屏幕亮起,为false代表当前屏幕熄灭
d.info.get( 'screenOn' )
# 仅点亮屏幕
d.screen_on()
# 点亮屏幕并解锁,注意如果有密码,则只能进入密码输入页面,需要输入密码才能解锁
d.unlock()
# 关闭屏幕
d.screen_off()
|
关于点击等操作(支持百分比):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
# 单击屏幕
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 ) 表示长按屏幕中心
|
当然还有其它的一些功能,例如向上滑动屏幕,直到指定文字出现为止:
1
|
d(scrollable = True ).scroll.to(text = "3年级2班" )
|
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注服务器之家的更多内容!
原文链接:https://blog.csdn.net/weixin_40901068/article/details/121242489