首頁>技術>

這篇文章主要介紹了python中spy++的使用超詳細教程,本文給大家介紹得非常詳細,對大家的學習或工作具有一定的參考借鑑價值,需要的朋友可以參考下

1、spy++的基本操作

指令碼之家下載spy++:

Microsoft Spy++ V15.0.26724.1 簡體中文綠色版 64位

1.1 視窗屬性查詢

拖住中間的“尋找工具”放到想要定位的軟體上,然後鬆開

1.2 視窗spy++定位2、python結合spy++

匯入必要的庫

import win32con

from win32 import win32gui

from win32 import win32clipboard as w

import pyautogui

import sys

2.1 獲取視窗全部屬性

def show_window_attr(hwnd):

"""

顯示視窗的屬性

:param hwnd: 視窗控制代碼(十進位制)

:return: 所有的屬性

WindowName: 視窗標題

ClassName: 視窗類名

HwndPy: 視窗控制代碼(十進位制)

HwndSpy: 視窗控制代碼(十六進位制)

"""

if not hwnd:

return

WindowName = win32gui.GetWindowText(hwnd)

ClassName = win32gui.GetClassName(hwnd)

HwndPy = hwnd

HwndSpy = hex(hwnd)

return (WindowName, ClassName, HwndPy, HwndSpy)

>>> int(0x31510)

202000

>>> show_window_attr(202000)

2.2 獲取全部頂層視窗

def show_top_windows():

"""

列出所有的頂級視窗及屬性

:return: 全部的頂層視窗及對應屬性

"""

hwndList = []

win32gui.EnumWindows(lambda hwnd, param: param.append(show_window_attr(hwnd)), hwndList)

return hwndList

獲取到的是整個桌面所有的視窗

>>> show_top_windows()

[('CClipboardThread', 'CClipboardThread', 2165890, '0x210c82'), ('G', 'GDI+ Hook Window Class', 463410, '0x71232'), ('', 'ForegroundStaging', 66338, '0x10322'), ('', 'ForegroundStaging', 66294, '0x102f6'), ('', 'tooltips_class32', 66234, '0x102ba'), ('', 'tooltips_class32', 66204, '0x1029c'), ('', 'tooltips_class32', 66200, '0x10298'), ('', 'tooltips_class32', 66196, '0x10294'), ('', 'tooltips_class32', 66192, '0x10290'), ('', 'tooltips_class32', 66174, '0x1027e'), ('', 'tooltips_class32', 66166, '0x10276'), ('', 'tooltips_class32', 66154, '0x1026a'), ('', 'tooltips_class32', 66888, '0x10548'), ('', 'tooltips_class32', 131762, '0x202b2'), ('', 'Q360NetmonClass', 197502, '0x3037e'), ('', 'tooltips_class32', 66208, '0x102a0'), ('', 'tooltips_class32', 11404742, '0xae05c6'), ('', 'tooltips_class32', 66214, '0x102a6'), ('', 'tooltips_class32', 66228, '0x102b4'), ('', 'tooltips_class32', 66222, '0x102ae'),

2.3 模糊查詢主窗體

def FindFuzzyTopWindow(FuzzyWindowName=None):

"""

根據標題模糊查詢全部符合條件的主窗體

:param FuzzyWindowName: 視窗標題部分文字

:return:

"""

all_windows = show_top_windows()

result = []

for window in all_windows:

if FuzzyWindowName in window[0]:

result.append(window)

return result

有時候我們需要透過“部分名稱”來找出主窗體的屬性找出了窗體標題含有“同花順”的全部窗體

>>> FindFuzzyTopWindow(FuzzyWindowName='同花順')

[('同花順(v8.80.80) - 我的板塊', 'Afx:400000:b:10003:6:d60ca7',

134184, '0x20c28'), ('同花順股靈通', '#32770', 462652, '0x70f3c')]

2.4 獲取目標父窗體下的全部子窗體

def FindSubHandles(pHandle=None, ClassName=None, WinName=None, index=None):

"""

返回窗體下全部的子窗體,預設主窗體下的窗體

:param pHandle: 視窗控制代碼(十進位制)

:param ClassName: 視窗類名,返回特定類名

:param WinName: 視窗標題,返回特定標題

:param index: 位置,返回特定位置的視窗

:return: 包含屬性的全部子視窗

"""

num = 0

handle = 0

SubHandlesList = []

while True:

# find next handle, return HwndPy

handle = win32gui.FindWindowEx(pHandle, handle, ClassName, WinName)

if handle == 0:

# no more handle

break

# get handle attribution

attr = show_window_attr(handle)

# append to list

SubHandlesList.append(tuple(list(attr) + [num]))

num += 1

if index is not None:

return SubHandlesList[index]

else:

return SubHandlesList

以剛剛“同花順”為例,十進位制控制代碼為134184其他引數在特定場合下會起作用

>>> FindSubHandles(pHandle=134184)

[('', 'Button', 69090, '0x10de2', 0),

('', 'ToolbarWindow32', 69272, '0x10e98', 1),

('', 'msctls_statusbar32', 265490, '0x40d12', 2),

('', 'msctls_statusbar32', 134664, '0x20e08', 3),

('', 'AfxFrameOrView42s', 134212, '0x20c44', 4),

('', 'AfxControlBar42s', 134180, '0x20c24', 5),

('', 'AfxControlBar42s', 134192, '0x20c30', 6),

('功能樹', 'AfxControlBar42s', 134194, '0x20c32', 7),

('', 'AfxControlBar42s', 134196, '0x20c34', 8),

('HqEmbededTradeContainer', 'Afx:400000:0', 69270, '0x10e96', 9),

('功能樹', 'Afx:400000:8:10003:10006e:0', 69320, '0x10ec8', 10),

('', 'Afx:400000:0', 69430, '0x10f36', 11),

('', 'Afx:400000:0', 69432, '0x10f38', 12)]

對比SPY++中的結果完全一致

2.5 獲取某個父視窗

下面全部的子視窗,遍歷所有視窗 這裡獲取到的是全部層級的子視窗

def ShowAllHandle(pHandle=None, HandleList=[[None]], HandleDict=dict()):

"""

生成視窗全部對應的關係

:param pHandle: 目標父視窗

:param HandleList: 預設為[[None]]

:param HandleDict: 用於存放對應關係

:return: 返回目標視窗下全部子父視窗的字典

"""

sys.setrecursionlimit(1000000)

if pHandle:

HandleList[-1][0] = pHandle

handles = FindSubHandles(HandleList[-1][0][2])

else:

handles = FindSubHandles()

for handle in handles:

HandleDict[handle] = pHandle

del HandleList[-1][0]

# 如果有葉節點,非空,則加入新的葉節點

if handles:

HandleList.append(handles)

HandleList = [HandleGroup for HandleGroup in HandleList if HandleGroup]

# 如果還有根就繼續遍歷,否則輸出樹

if HandleList:

return ShowAllHandle(pHandle=HandleList[-1][0], HandleList=HandleList, HandleDict=HandleDict)

else:

return HandleDict

pHandle:這個引數設定為

('同花順(v8.80.80) - 我的板塊', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28')

呼叫函式

ShowAllHandle(FindFuzzyTopWindow(FuzzyWindowName='同花順')[0])

得到一個全部對應關係的字典

2.6 找到特定視窗的路徑

因為控制代碼在不同電腦上的結果是不同的,所以我們要找到這個唯一的路

def FindHandlePath(TargetHandle, num):

"""

尋找特定視窗的尋找路徑

:param TargetHandle: 視窗控制代碼(十六進位制)

:param num: 視窗所屬index,在spy++內檢視

:return:

ParentWindow:頂層視窗

TargetPath:路徑的index

"""

AllPath = ShowAllHandle(pHandle=None, HandleList=[[None]], HandleDict=dict())

key = tuple(list(show_window_attr(int(TargetHandle))) + [num])

handlepath = [key]

while True:

key = AllPath[key]

if not key:

handlepath = handlepath[::-1]

ParentWindow = handlepath[0]

TargetPath = [(i[-1]) for i in handlepath[1:]]

return ParentWindow, TargetPath

handlepath.append(key)

邏輯是找到全部層級的對應關係,然後反向搜尋以“同花順”地“買入”按鈕為例,透過SPY++查詢

我們得到其十六進位制控制代碼為“0x40D98”,同時index為6

TargetHandle=0x40D98, num=6獲取路徑如下

>>> FindHandlePath(TargetHandle=0x40D98, num=6)

(('同花順(v8.80.80) - 我的板塊', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28', 308),

[5, 0, 6])

主窗體:同花順(v8.80.80) - 我的板塊5:第6個子窗體0:第1個子窗體6:第7個子窗體2.7 根據路徑來查詢某個特定視窗的控制代碼

def FindTargetHandle(pHandle, WindowList):

"""

遞迴尋找子視窗的控制代碼

:param pHandle: 祖父視窗的完整控制代碼 (WindowName, ClassName, HwndPy, HwndSpy)

:param WindowList: 子視窗列表

:return: 目標視窗的完整屬性

"""

for i in range(len(WindowList)):

pHandle = FindSubHandles(pHandle[2], index=WindowList[i])

return pHandle

現在我們已經有了某個窗體的查詢路徑,透過上述函式來查詢其控制代碼我們的路徑是:(('同花順(v8.80.80) - 我的板塊', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28', 308), [5, 0, 6])呼叫函式

>>> FindTargetHandle(('同花順(v8.80.80) - 我的板塊', 'Afx:400000:b:10003:6:d60ca7', 134184, '0x20c28', 308), [5, 0, 6])

('', 'Button', 265624, '0x40d98', 6)

獲得到了名為“Button”的窗體對比我們剛剛的結果,是正確的。 2.8 根據控制代碼定位窗體這個按鈕的十進位制控制代碼為:265624呼叫函式GetWindowRect

>>> x,y,m,n = win32gui.GetWindowRect(265624)

>>> pyautogui.moveTo((x+m)/2, (y+n)/2)

此時滑鼠會自動移動到這個窗體上

參考連結:https://blog.csdn.net/qq_25408423/article/details/80884114https://blog.csdn.net/seele52/article/details/17504925

Spy++的使用方法及下載教程

下載地址:

https://www.jb51.net/softs/602474.html

很多朋友都對視窗控制代碼比較迷糊,這篇短文就以spy++這個軟體為主,介紹下視窗控制代碼和使用按鍵外掛時,如果對這個控制代碼傳送訊息,即所謂的後臺掛機。spy++這個軟體來自VC++,裝好VC後,就可以在工具中看到。

2使用spy++的查詢功能

3點開後的效果

4我們把黑色準星拖動到記事本窗體上

這三個控制代碼從上到下依次是:記事本的主控制代碼文字輸入框的控制代碼狀態列的控制代碼我們要給記事本傳送按鍵訊息,就要選對傳送物件(控制代碼),這裡顯然,我們要傳送給“文字輸入框”。我們雙擊“文字輸入框”這裡,可以看到“視窗標題”、“視窗大小”、“視窗位置”等資訊。點選“類”選項卡,檢視下類名是什麼,程式碼裡需要類名做引數

這裡的控制代碼,類名在Python-pywin32是有用的

18
  • BSA-TRITC(10mg/ml) TRITC-BSA 牛血清白蛋白改性標記羅丹明
  • 架構解密從分散式到微服務:古老又有生命力的 RPC