最近和跟着同事一起玩阴阳师,发现这个游戏有太多重复操作了,这完全就是浪费生命啊;所以想到用python写一个自动挂机脚本。
最开始想得很简单,就是一直去找相应得按钮,然后点击就可以了。所以直接用pyautogui的图片定位和点击功能就行了,也确实实现了,代码如下:
import pyautogui,time pyautogui.FAILSAFE = True '''PyAutoGUI提供了一个保护措施。当pyautogui.FAILSAFE = True时,如果把鼠标光标在屏幕左上角, PyAutoGUI函数就会产生pyautogui.FailSafeException异常,用于在程序失控时退出''' time.sleep(2) def get_point(picture): '''精确匹配某个按钮的位置;通过传入图片获取图片在屏幕上的定位,一旦获取到值则退出,否则继续尝试''' picture = './img/' + picture count = 5 while count > 0: point = pyautogui.locateCenterOnScreen(picture) if point is not None: return point else: count -= 1 def get_range(picture): '''用模糊匹配得到某个按钮的大概位置,一般是在那种点击屏幕任意位置的情况,或只需要知道某个按钮在不在当前屏幕的情况使用 要动的图片采用模糊匹配,否则精确匹配的话,图片一直在动像素也在变化,就不能定位到了''' picture = './img/' + picture count = 5 while count > 0: range = pyautogui.locateCenterOnScreen(picture, grayscale=True, confidence=0.5) if range is not None: return range else: count -= 1 def click_button(picture,accurate=True): '''点击按钮的函数,默认精确度为True,即默认为精确点击,如果accurate=False,则为模糊点击,用于动态图形按钮''' if accurate==True: action = get_point(picture) else: action = get_range(picture) if action is not None: pyautogui.click(action,duration=0.5) def cycle_fight(): '''用户循环重复的战斗场景,如果刷觉醒材料,刷御魂''' while True: click_button('tiaozhan.PNG') click_button('zhunbei.PNG') click_button('over.PNG') cycle_fight()
最后发现这样似乎不科学,不仅慢还浪费资源;然后又想到在准备完成和战斗结束这段时间,有很长的空白期,完全可以让程序停止啊,所以又出现了下面的代码:
import pyautogui,time
pyautogui.FAILSAFE = True
'''PyAutoGUI提供了一个保护措施。当pyautogui.FAILSAFE = True时,如果把鼠标光标在屏幕左上角,
PyAutoGUI函数就会产生pyautogui.FailSafeException异常,用于在程序失控时退出'''
time.sleep(2)
def get_point(picture):
'''精确匹配某个按钮的位置;通过传入图片获取图片在屏幕上的定位,一旦获取到值则退出,否则继续尝试'''
picture = './img/' + picture
count = 5
while count > 0:
point = pyautogui.locateCenterOnScreen(picture)
if point is not None:
return point
else:
count -= 1
def get_range(picture):
'''用模糊匹配得到某个按钮的大概位置,一般是在那种点击屏幕任意位置的情况,或只需要知道某个按钮在不在当前屏幕的情况使用
要动的图片采用模糊匹配,否则精确匹配的话,图片一直在动像素也在变化,就不能定位到了'''
picture = './img/' + picture
count = 5
while count > 0:
range = pyautogui.locateCenterOnScreen(picture, grayscale=True, confidence=0.5)
if range is not None:
return range
else:
count -= 1
def click_button(picture,accurate=True):
'''点击按钮的函数,默认精确度为True,即默认为精确点击,如果accurate=False,则为模糊点击,用于动态图形按钮'''
if accurate==True:
action = get_point(picture)
else:
action = get_range(picture)
if action is not None:
pyautogui.click(action,duration=0.5)
return True #点击成功则返回True
else:
return None #通过这个返回值判定有没有点击成功
def on_fight(sec):
'''这个函数用于模拟从【准备】到【战斗结束】这段时间,在这段时间里程序应该是阻塞的,这样就不用一直去找对应的按钮,从而消耗大量的系统资源'''
fight_run = click_button('zhunbei.PNG') #点击准备按钮则战斗开始
if fight_run is not None: #如果成功点击了准备按钮则代表战斗开始,程序进入寻找【战斗结束按钮】的状态
count = 0
while True:
time.sleep(sec) #设定每几秒检测一次战斗是否结束,这个值可以具体情况设置
fight_over = click_button('over.PNG')
if fight_over is not None:
return True
else:
count +=1 #记录循环次数
if count*sec > 600:
'''用循环次数乘以中断时间,大约等于战斗过程的时间,这里的意思是战斗过程大于10分钟,一般这种情况,肯定
是在点击了准备之后,战斗过程中异常中断,这时候程序会一直陷入这个寻找战斗结束的死循环中,但这是没有意义的,
所以直接退出整个程序'''
exit(1)
else:
return None #如果没有点击【准备】按钮,则返回为空,继续进入下一次寻找进入战斗起始按钮的过程
def cycle_fight(sec):
'''用户循环重复的战斗场景,如果刷觉醒材料,刷御魂'''
while True:
click_button('tiaozhan.PNG')
on_fight(sec)
def story_task():
'''用于过废话连篇的剧情'''
while True:
click_button('tiaoguo.PNG')
click_button('dialogue.PNG')
click_button('storyJump.PNG')
def explore_task(sec):
'''用于过探索副本'''
while True:
click_button('fight.PNG',accurate=False)
click_button('masterFight.PNG',accurate=False)
on_fight(sec)
#explore_task()
# cycle_fight(10)
story_task()
这样一来虽然降低了系统资源消耗但代码逻辑变得极为复杂,然后便想到用多线程封装,之后效率确实极大的提升了:
import pyautogui,time
import threading
pyautogui.FAILSAFE = True
'''PyAutoGUI提供了一个保护措施。当pyautogui.FAILSAFE = True时,如果把鼠标光标在屏幕左上角,
PyAutoGUI函数就会产生pyautogui.FailSafeException异常,用于在程序失控时退出'''
time.sleep(2)
class FindButton(threading.Thread):
def __init__(self, picture):
super(FindButton, self).__init__()
self.picture = './img/' + picture
def run(self):
while True:
self.point = pyautogui.locateCenterOnScreen(self.picture, confidence=0.8)
if self.point is not None:
pyautogui.click(self.point, duration=0.5)
challenge = FindButton('tiaozhan.PNG')
prepare = FindButton('zhunbei.PNG')
over = FindButton('over.PNG')
challenge.start()
prepare.start()
over.start()
但是这样一来,系统资源的消耗也成倍的增加了,相当于每多一个点击操作,就要多使用一倍的系统资源。那不如用协程来解决吧,既是单线程,又可以异步进行;
最后证明协程比多线程稍慢(毕竟协程是需要排队的,协程直接的切换也是需要消耗时间的),比单线程则是快了太多了
而且消耗的系统资源也极大的降低了(甚至比单线程的情况还低),并且代码也简单了太多了!
import gevent import pyautogui def click(picture): picture = './img/' + picture while True: point = pyautogui.locateCenterOnScreen(picture, confidence=0.8) if point is None: gevent.sleep(1) else: pyautogui.click(point, duration=0.5) def cycle_fight(): '''用于循环重复的战斗场景,如果刷觉醒材料,刷御魂''' gevent.joinall([ #利用joinall方法将每一步操作加入协程池中 gevent.spawn(click,'tiaozhan.PNG'), #每一个协程的加入方法是:(函数名,参数) gevent.spawn(click,'zhunbei.PNG'), gevent.spawn(click,'over.PNG') ]) def story_task(): ''''用于过废话连篇的剧情任务''' gevent.joinall([ gevent.spawn(click, 'dialogue.PNG'), gevent.spawn(click, 'tiaoguo.PNG'), gevent.spawn(click, 'storyJump.PNG'), gevent.spawn(click, 'fight.PNG'), gevent.spawn(click, 'zhunbei.PNG'), gevent.spawn(click, 'over.PNG') ]) def explore_task(): '''用于过探索副本''' gevent.joinall([ gevent.spawn(click, 'fight.PNG'), gevent.spawn(click, 'masterFight.PNG'), gevent.spawn(click, 'zhunbei.PNG'), gevent.spawn(click, 'over.PNG') ]) cycle_fight()