最近老师布置了一个作业,要求我们建立一个6位数字的gmail账号,然而我脸太黑,试了好多次都不成功,于是决定用python来写一个脚本遍历输入所有6位数。这就要求掌握使用python捕捉和模拟键盘操作了,于是我查阅了网上诸多资料,终于实现了这个功能,特此记录下来方便以后复习。
使用的python版本为:3.6.1,使用anaconda配置的python环境
参考博文
首先安装pyautogui库:
conda install -c jim-hart pyautogui
PyAutoGUI使用x,y坐标,屏幕左上角坐标是(0, 0)
获取屏幕分辨率
import pyautogui as pag
screenWidth, screenHeight = pag.size()
截屏功能
# 返回一个Pillow/PIL的Image对象
img=pag.screenshot()
img.save('foo.png')
pag.screenshot('foo.png')
获得某个坐标的像素
(r,g,b)=img.getpixel((50, 200))
# (30, 132, 153)
# 判断屏幕坐标的像素是不是等于某个值
ifEqual=pag.pixelMatchesColor(50, 200, (30, 132, 153))
#True
如果你不需要截取整个屏幕,还有一个可选的region参数。你可以把截取区域的左上角XY坐标值和宽度、高度传入截取
img = pag.screenshot(region=(0, 0, 300 ,400))
在屏幕上找到按钮所在的坐标。其实很简单,首先对你要点击的按钮截个图,就叫button.png吧。然后使用locateOnScreen函数找到按钮所在的位置
pag.locateOnScreen('button.png')
# (643, 745, 70, 29)
locateOnScreen其实就是简单的颜色对比,如果有一个像素不匹配,它就会返回None。这个函数返回了匹配图形的坐标,找到中间点:
x, y = pag.center((643, 745, 70, 29)) # 获得中心点
pyautogui.click(x, y)
要检查XY坐标是否在屏幕上,需要用onScreen()函数来检验,如果在屏幕上返回True:
pag.onScreen(0, 0) #True
pag.onScreen(0, -1) #False
获取当前鼠标的坐标
currentMouseX, currentMouseY = pag.position()
鼠标点击
pag.click()
使用click()函数发送虚拟鼠标点击,默认情况下在鼠标所在的位置点击左键。函数原型:
pag.click(x=cur_x, y=cur_y, button='left')
x,y是要点击的位置,默认是鼠标当前位置
button是要点击的按键,有三个可选值:‘left’, ‘middle’, ‘right’
每个按键按下和松开两个事件可以分开处理:
pag.mouseDown(x=moveToX, y=moveToY, button='left')
pag.mouseUp(x=moveToX, y=moveToY, button='left')
双击
pag.doubleClick()
右击
pag.rightClick()
中击
pag.middleClick()
鼠标移动
pag.moveTo(x,y,duration) # 绝对坐标
pag.moveRel(x,y,duration) # 相对坐标
鼠标拖拽
注意:duration时间不能太短,拖动太快有些系统会吃不消
pag.dragTo(x,y,duration)
pag.dragRel(x,y,duration)
多次点击
可以设置clicks参数,还有interval参数可以设置每次单击之间的时间间隔。例如:
# 双击左键
pyautogui.click(clicks=2)
# 两次单击之间停留0.25秒
pyautogui.click(clicks=2, interval=0.25)
# 三击右键
pyautogui.click(button='right', clicks=2, interval=0.25)
滚轮
使用函数scroll(),它只接受一个整数。如果值为正往上滚,值为负往下滚。
pag.scroll(200)
缓动/渐变(Tween / Easing)函数
缓动/渐变函数的作用是让光标的移动更炫。如果你不需要用到的话,你可以忽略这些
缓动/渐变函数可以改变光标移动过程的速度和方向。通常鼠标是匀速直线运动,这就是线性缓动/渐变函数。PyAutoGUI有30种缓动/渐变函数,可以通过pyautogui.ease*?查看。其中,pyautogui.easeInQuad()函数可以用于moveTo(),moveRel(),dragTo()和dragRel()函数,光标移动呈现先慢后快的效果,整个过程的时间还是和原来一样。而pyautogui.easeOutQuad函数的效果相反:光标开始移动很快,然后慢慢减速。pyautogui.easeOutElastic是弹簧效果,首先越过终点,然后再反弹回来。例如:
# 开始很慢,不断加速
pyautogui.moveTo(100, 100, 2, pyautogui.easeInQuad)
# 开始很快,不断减速
pyautogui.moveTo(100, 100, 2, pyautogui.easeOutQuad)
# 开始和结束都快,中间比较慢
pyautogui.moveTo(100, 100, 2, pyautogui.easeInOutQuad)
# 一步一徘徊前进
pyautogui.moveTo(100, 100, 2, pyautogui.easeInBounce)
# 徘徊幅度更大,甚至超过起点和终点
pyautogui.moveTo(100, 100, 2, pyautogui.easeInElastic)
输入字符串
pag.typewrite('Hello world')
上面的字符串是一次输入,为了唬人可以延迟输入
pag.typewrite('Hello world!', 0.25)
PyAutoGUI键盘表:
字符串 | 代表按键 |
---|---|
‘enter’(或‘return’ 或 ‘\n’) | 回车 |
‘esc’ | ESC键 |
‘shiftleft’, ‘shiftright’ | 左右SHIFT键 |
‘altleft’, ‘altright’ | 左右ALT键 |
‘ctrlleft’, ‘ctrlright’ | 左右CTRL键 |
‘tab’ (‘\t’) | TAB键 |
‘backspace’, ‘delete’ | BACKSPACE 、DELETE键 |
‘pageup’, ‘pagedown’ | PAGE UP 和 PAGE DOWN键 |
‘home’, ‘end’ | HOME 和 END键 |
‘up’, ‘down’, ‘left’,‘right’ | 箭头键 |
‘f1’, ‘f2’, ‘f3’…. | F1…….F12键 |
‘volumemute’, ‘volumedown’,‘volumeup’ | 有些键盘没有 |
‘pause’ | PAUSE键 |
‘capslock’, ‘numlock’,‘scrolllock’ | CAPS LOCK, NUM LOCK, 和 SCROLLLOCK 键 |
‘insert’ | INS或INSERT键 |
‘printscreen’ | PRTSC 或 PRINT SCREEN键 |
‘winleft’, ‘winright’ | Win键 |
‘command’ | Mac OS X command键 |
消息弹窗函数
pyautogui.alert('这个消息弹窗是文字+OK按钮')
pyautogui.confirm('这个消息弹窗是文字+OK+Cancel按钮')
pyautogui.prompt('这个消息弹窗是让用户输入字符串,单击OK')
#返回用户输入的字符串,如果用户什么都不输入,则返回None
Python移动鼠标、点击键盘非常快,有可以导致其他应用出现问题。在这种情况下,程序可能会失控(即使是按照你的意思执行的),那时就需要中断。如果鼠标还在自动操作,就很难在程序窗口关闭它。
为了能够及时中断,PyAutoGUI提供了一个保护措施。当pyautogui.FAILSAFE = True时,如果把鼠标光标在屏幕左上角,PyAutoGUI函数就会产生pyautogui.FailSafeException异常。如果失控了,需要中断PyAutoGUI函数,就把鼠标光标在屏幕左上角。要禁用这个特性,就把FAILSAFE设置成False
pag.FAILSAFE = False
通过把pyautogui.PAUSE设置成float或int时间(秒),可以为所有的PyAutoGUI函数增加延迟。默认延迟时间是0.1秒。在函数循环执行的时候,这样做可以让PyAutoGUI运行的慢一点,非常有用。例如:
pag.PAUSE = 2.5
pag.moveTo(100,100);
pag.click()
经过一晚上的奋战,终于找到了一个6位数的gmail邮箱名,成功把作业提交了,脚本代码如下
# -*- coding: utf-8 -*-
""" Created on Sat Sep 23 00:23:10 2017 @author: WangTong """
import pyautogui as pag
import time
pag.FAILSAFE = True
output=open('C:\\Users\\WangTong\\Desktop\\寻找gmail\\output.txt','w+')
pag.moveTo(100,800)
pag.scroll(100)
pag.moveTo(500,1720)#右移
pag.dragTo(1000,1720,0.5)
try:
for num in range(800000,1000000):
flag=True
pag.moveTo(1057,754)#移动到选择用户名窗口
pag.click()
pag.hotkey('ctrl','a')
pag.press('delete')
pag.typewrite(str(num))
pag.moveTo(800,754)#移动到选择用户名窗口
pag.click()
time.sleep(1)
img = pag.screenshot(region=(540, 440, 10,1))
for i in range(1,10):
(r,g,b)=img.getpixel((i,0))
if g<100:
flag=False# 出现了红框,说明该用户名已经有人用了
break
if flag:
#之前可能是网速较慢,延时5s再判断一次
flag=True
pag.click()
time.sleep(5)
img = pag.screenshot(region=(540, 440, 10,1))
for i in range(1,10):
(r,g,b)=img.getpixel((i,0))
if g<100:
flag=False
break
if flag:
#这个密码可以使用,但也存在google挂掉的情况,无论如何,先注册试试
pag.moveTo(1057,870)#设置密码
pag.click()
pag.hotkey('ctrl','a')
pag.press('delete')
pag.typewrite('dasdhasiduha')
pag.moveTo(1057,990)#确认密码
pag.click()
pag.hotkey('ctrl','a')
pag.press('delete')
pag.typewrite('dasdhasiduha')
pag.moveTo(1450,1680)#下一步
pag.click()
pag.moveTo(920,1520)#按键
pag.click()
pag.moveTo(1250,1580)#我同意
pag.click()
time.sleep(4)
img = pag.screenshot(region=(160, 450, 1 ,1))
(r,g,b)=img.getpixel((0, 0))
#可以(74, 140, 246)
#不可以(255, 255, 255)
if r<150:#注册失败
print(num,'可以')
pag.moveTo(40,100)#返回上一层
pag.click()
time.sleep(1)
pag.scroll(100)
pag.moveTo(500,1720)#右移
pag.dragTo(1000,1720,0.5)
output.write(str(num)+':Ok')
output.write('\n')
else:#注册成功
print(num,'不可以')
pag.scroll(100)
pag.moveTo(500,1720)#右移
pag.dragTo(1000,1720,0.5)
output.write(str(num)+':No')
except:
print('error')
finally:
output.close()
总体思路就是遍历所有6位数用户名,如果该用户名已经被人注册了,用户名输入框会变成红色提示,如果没有被人注册,就是灰色输入框,所以我就依靠输入框的像素值来判断该用户名是否可以使用。不过有时候会因为网速问题加上gmail注册界面的一些莫名其妙的问题,一个账号明明已经被人注册了,但是输入进去之后不会有任何提示,必须要点击下一步才能提示你已经被人注册了,所以我就把完整的注册流程给模拟了一边,并且找了注册成功和注册失败界面中的某一个不一样的像素点作为区分,如果注册失败了,界面不会跳转,只要继续循环过程即可,如果注册成功了,为了保险起见,我将这个可以注册的账号输出的output.txt文件里,然后返回上一个界面继续尝试下一个数字。不过要注意的一点就是,无论注册成功还是失败,界面的位置都会变化,这个时候一定要将界面调整到右上角,不然之前设定的像素点坐标就错了。最后我跑了一个中午,找到了两个可以使用的数字,第一个不知道因为什么原因已经被人注册了但是误判为没有被注册,但是第二个数字可以使用,于是我的作业就完成了。
Ps:好多同学随手一试就找到了一个可以使用的用户名,我用脚本试了两千多个数字才找到一个,这就是欧皇和非酋的区别了吧/(ㄒoㄒ)/~~