12.3 实现模拟鼠标录制回放
本节将向读者介绍如何使用键盘鼠标操控模拟技术,键盘鼠标操控模拟技术是一种非常实用的技术,可以自动化执行一些重复性的任务,提高工作效率,在Windows系统下,通过使用各种键盘鼠标控制函数实现动态捕捉和模拟特定功能的操作。
有时我们经常需要进行重复性的鼠标操作,例如繁琐的点击、拖拽。这些任务可能消耗大量时间和精力,为了解决这个问题,可自行设计并实现一个简单而强大的鼠标录制回放工具,旨在帮助用户轻松录制鼠标动作,通过借助鼠标录制回放工具,用户可以轻松实现自动化操作,从而解放双手。
首先我们需要创建一个Write_File
函数,当用户每次调用该函数时都会向特定的文件内追加写入一条记录,此外还需要增加一个split
函数,该函数用于将特定的一条记录根据特定的分隔符切割,保留分隔符后面的坐标信息。
// 切割字符串
int split(char dst[][32], char* str, const char* spl)
{int n = 0;char* result = NULL;result = strtok(str, spl);while (result != NULL){strcpy(dst[n++], result);result = strtok(NULL, spl);}return n;
}// 每次写入一行
int Write_File(char* path, char* msg)
{FILE* fp = fopen(path, "a+");if (fp == NULL) return -1;char ch, buffer[1024];int index = 0;while (msg[index] != '\0'){fputc(msg[index], fp);index++;}fclose(fp);return 1;
}
接着我们需要实现Recording()
函数部分,该函数的左右是用于捕捉当前鼠标坐标与点击事件,函数中通过调用GetCursorPos()
获取当前鼠标的屏幕坐标位置,这个函数参数传递非常简单,只需要传入一个POINT
类型的结构体变量,其函数原型如下所示;
BOOL GetCursorPos(LPPOINT lpPoint);
参数:
lpPoint
:指向POINT
结构的指针,用于接收鼠标的屏幕坐标位置。
返回值:
- 如果函数成功,返回值为非零,表示获取鼠标位置成功;
- 如果函数失败,返回值为零,表示获取鼠标位置失败。
POINT
结构包含了两个成员变量 x
和 y
,分别表示鼠标在屏幕上的横坐标和纵坐标。
当有了当前鼠标坐标位置以后,接着就是需要获取到鼠标点击事件,鼠标点击可使用GetAsyncKeyState
获取指定虚拟键码对应的键盘键的状态,该函数原型如下所示;
SHORT GetAsyncKeyState(int vKey);
参数:
vKey
:指定虚拟键码,它是一个整数,表示要获取的键的键码。
返回值:
- 如果指定的虚拟键处于按下状态,返回值的最高位(符号位)为 1,其余位表示次数(持续时间)。如果指定的虚拟键处于释放状态或者参数无效,返回值为 0。
GetAsyncKeyState
函数允许检测键盘中某个虚拟键的状态,无论这个虚拟键是否处于焦点的窗口中。它适用于各种应用,通过VK_LBUTTON
可用于检测鼠标左键是否被按下,通过VK_RBUTTON
则可用于检测鼠标右键状态。
代码的主要功能如下:
- 在
Recording
函数中,使用一个死循环不断检测鼠标的位置和按键状态。 - 使用
GetCursorPos
函数获取当前鼠标的位置,并将其保存在x
和y
变量中。 - 使用
GetAsyncKeyState
函数检测鼠标左键和右键的状态,并将其保存在lbutton
和rbutton
变量中。 - 如果当前的鼠标位置或按键状态与之前保存的值不同,表示鼠标动作发生了变化,将当前的位置和按键状态记录下来。
- 将记录的鼠标动作信息以字符串的形式写入脚本文件,格式为 “X:位置,Y:位置,L:左键状态,R:右键状态”。
- 保存当前的鼠标位置和按键状态,用于下一次循环时比较是否发生了变化。
// 录制脚本
void Recording(char *script)
{int static_x = 0, static_y = 0;bool static_lbutton = 0, static_rbutton = 0;while (1){POINT Position;GetCursorPos(&Position);int x = Position.x;int y = Position.y;bool lbutton = GetAsyncKeyState(VK_LBUTTON);bool rbutton = GetAsyncKeyState(VK_RBUTTON);if (x != static_x || y != static_y || lbutton != static_lbutton || rbutton != static_rbutton){char szBuf[1024] = { 0 };std::cout << "X轴 = " << x << " Y轴 = " << y << " 鼠标左键 = " << lbutton << " 鼠标右键 = " << rbutton << std::endl;sprintf(szBuf, "X:%d,Y:%d,L:%d,R:%d\n", x, y, lbutton, rbutton);Write_File((char*)script, szBuf);static_x = x;static_y = y;static_lbutton = lbutton;static_rbutton = rbutton;}}
}
接着我们继续封装Play()
回放功能,该功能的实现原理与录制保持一致,通过逐条读取传入文件中的参数,并调用SetCursorPos
实现鼠标位置的移动操作,该函数与获取参数传递保持一致,这里我们需要注意mouse_event
函数,该函数用于模拟鼠标的各种事件,如鼠标移动、鼠标按键的点击和释放等,其函数原型如下所示;
void mouse_event(DWORD dwFlags, DWORD dx, DWORD dy, DWORD dwData, ULONG_PTR dwExtraInfo);
其中dwFlags指定要模拟的鼠标事件类型和选项。可以是以下常量的组合;
MOUSEEVENTF_ABSOLUTE
:指定鼠标位置是绝对坐标。如果不设置此标志,则坐标是相对于当前鼠标位置的增量。MOUSEEVENTF_MOVE
:模拟鼠标移动事件。MOUSEEVENTF_LEFTDOWN
:模拟鼠标左键按下事件。MOUSEEVENTF_LEFTUP
:模拟鼠标左键释放事件。MOUSEEVENTF_RIGHTDOWN
:模拟鼠标右键按下事件。MOUSEEVENTF_RIGHTUP
:模拟鼠标右键释放事件。
其他常量可根据需要自行查阅相关文档。
dx
:鼠标事件发生时的横坐标(绝对坐标或增量坐标,根据dwFlags
决定)。dy
:鼠标事件发生时的纵坐标(绝对坐标或增量坐标,根据dwFlags
决定)。dwData
:鼠标事件的一些数据。对于滚轮事件,它表示滚动的数量。对于其他事件,通常设为 0。dwExtraInfo
:额外的信息。通常设为 0。
mouse_event
函数允许模拟鼠标事件,通过设置 dwFlags
参数来指定需要模拟的事件类型,设置 dx
和 dy
参数来指定事件发生时的鼠标位置。通过调用这个函数,可以实现自动化鼠标操作,如模拟鼠标点击、拖动等。
如下代码段实现了鼠标动作脚本的回放功能,它从之前录制保存的脚本文件中读取鼠标动作信息,并按照脚本中记录的顺序模拟鼠标动作,实现鼠标动作的回放。
代码的主要功能如下:
- 在
Play
函数中,打开之前保存的脚本文件,并使用fgets
函数每次读取一行数据,保存在buf
字符数组中。 - 使用
split
函数切割每行数据,将每行数据切割成以逗号分隔的四个字符串,并将这四个字符串转换为整数类型保存在key_item
数组中。 - 根据
key_item
数组中的数据,判断是否需要进行鼠标点击动作,并调用mouse_event
函数模拟鼠标点击。 - 调用
SetCursorPos
函数设置鼠标的位置,并使用Sleep
函数模拟鼠标移动的延时,实现鼠标动作的回放。 - 循环执行以上步骤,直到脚本文件中的所有动作都被回放完毕。
// 回放脚本
void Play(char *script)
{FILE* fp = fopen(script, "r");char buf[1024];while (feof(fp) == 0){// 每次读入一行memset(buf, 0, 1024);fgets(buf, 1024, fp);// 以逗号切割char split_comma[4][32] = { 0 };int comma_count = split(split_comma, buf, ",");int key_item[4] = { 0 };// std::cout << "长度: " << comma_count << std::endl;for (int x = 0; x < comma_count; x++){// 继续切割冒号char split_colon[2][32] = { 0 };split(split_colon, split_comma[x], ":");// std::cout << "字典键 = " << split_colon[0] << " 字典值 = " << split_colon[1] << std::endl;key_item[x] = atoi(split_colon[1]);}if (key_item[3] != 0){mouse_event(MOUSEEVENTF_LEFTUP | MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);}if (key_item[4] != 0){mouse_event(MOUSEEVENTF_RIGHTUP | MOUSEEVENTF_RIGHTDOWN, 0, 0, 0, 0);}// 得到数据后开始回放SetCursorPos(key_item[0], key_item[1]);Sleep(70);}
}
最后是主函数部分,我们通过RegisterHotKey
函数注册两个全局热键,通过F1
实现鼠标录制部分,通过F2
则实现鼠标回放,最后通过GetMessage
函数接收全局消息事件,当出现WM_HOTKEY
消息则依次判断是否启用录制回放等功能,代码如下所示;
int main(int argc, char* argv[])
{// 注册热键 F1 , F2if (0 == RegisterHotKey(NULL, 1,0, VK_F1)){cout << GetLastError() << endl;}if (0 == RegisterHotKey(NULL, 2,0, VK_F2)){cout << GetLastError() << endl;}if (0 == RegisterHotKey(NULL, 3, 0, VK_F3)){cout << GetLastError() << endl;}// 消息循环MSG msg = { 0 };while (GetMessage(&msg, NULL, 0, 0)){switch (msg.message){case WM_HOTKEY:{if (1 == msg.wParam){std::cout << "录制脚本" << std::endl;Recording((char *)"d://script.txt");}else if (2 == msg.wParam){std::cout << "回放脚本" << std::endl;Play((char *)"d://script.txt");}else if (3 == msg.wParam){exit(0);return 0;}break;}default:break;}}return 0;
}
读者可自行编译并运行这段代码,通过录制一段鼠标功能并回放,输出效果图如下所示;
本文作者: 王瑞
本文链接: https://www.lyshark.com/post/d3a88993.html
版权声明: 本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
相关文章:

12.3 实现模拟鼠标录制回放
本节将向读者介绍如何使用键盘鼠标操控模拟技术,键盘鼠标操控模拟技术是一种非常实用的技术,可以自动化执行一些重复性的任务,提高工作效率,在Windows系统下,通过使用各种键盘鼠标控制函数实现动态捕捉和模拟特定功能的…...

【计算机网络-自顶向下方法】应用层(SMTP、POP3、DNS)
目录 1. Electronic Mail电子邮件应用画像1.1 电子邮件系统1.2 邮件报文格式1.3 邮件访问 2. DNS(Domain Name System)2.1 DNS提供的服务2.2 DNS工作机理2.3 DNS资源记录2.4 DNS协议,报文2.5 小结 1. Electronic Mail 电子邮件应用画像 应用…...

【Pm4py第八讲】关于Statistics
本节用于介绍pm4py中的统计函数,包括统计轨迹变体、案例持续时间、案例到达时间等。 1.函数概述 本次主要介绍Pm4py中一些常见的统计函数,总览如下表: 函数名说明pm4py.stats.get_start_activities()从事件日志中获取开始活动。pm4py.stats.…...

【Azure 架构师学习笔记】-Azure Data Factory (5) --Data Flow
本文属于【Azure 架构师学习笔记】系列。 本文属于【Azure Data Factory】系列。 接上文【Azure 架构师学习笔记】-Azure Data Factory (4)-触发器详解-事件触发器 前言 Azure Data Factory, ADF 是微软Azure 的ETL 首选服务之一, 是Azure data platfor…...
uniapp之ios开发及支付整体流程爬坑记录
前言 在写这篇记录的时候,关于ios的支付已经对接的差不多了,下一步就是测试好了直接发版,总共花了好几周的时间,从0到1对于首次做ios支付来说,确实很多坑。 其实业务层面很简单,甚至比安卓支付还简单&…...

AutoDL百川大模型体验
文章目录 镜像克隆模型下载测试效果AutoDL自定义服务 感谢AutoDL和CodeWithGPU这两个平台,让我们能低成本,低门槛地部署体验这些大模型 镜像克隆 我是在CodeWithGPU上克隆的这个镜像 模型下载 codewithgpu有介绍 注意这三个文件都需要下载 把那个&quo…...

蓝桥杯每日一题2023.10.8
题目描述 七段码 - 蓝桥云课 (lanqiao.cn) 题目分析 所有的情况我们可以分析出来一共有2的7次方-1种,因为每一个二极管都有选择和不选择两种情况,有7个二极管,但是还有一种都不选的情况需要排除,故-1 枚举每个方案看是否符合要…...

jmeter,性能测试,Locust
一。性能测试的概念 1.性能:就是软件质量属性中的 “ 效率 ” 特性 2.效率特性: 时间特性:指系统处理用户请求的响应时间 资源特性:指系统在运行过程中,系统资源的消耗情况 CPU 内存 磁盘IO(磁盘的写…...

opencv图像的直方图,二维直方图,直方图均衡化
文章目录 opencv图像的直方图,二维直方图,直方图均衡化一、图像的直方图1、什么是图像的直方图:2、直方图的作用:3、如何绘制图像的直方图:(1)cv::calcHist()函数原型:英文单词 calc…...

c++中的map和set
文章目录 1. 关联式容器2. 键值对3. 树形结构的关联式容器3.1 set3.1.1 set的介绍3.1.2 set的使用 3.2 map3.2.1 map的介绍3.2.2 map的使用 3.3 multiset3.3.1 multiset的介绍3.3.2 multiset的使用 3.4 multimap3.4.1 multimap的介绍3.4.2 multimap的使用 1. 关联式容器 在初阶…...

Swagger使用详解
目录 一、简介 二、SwaggerTest项目搭建 1. pom.xml 2. entity类 3. controller层 三、基本使用 1. 导入相关依赖 2. 编写配置文件 2.1 配置基本信息 2.2 配置接口信息 2.3 配置分组信息 2.3.1 分组名修改 2.3.2 设置多个分组 四、常用注解使用 1. ApiModel 2.A…...

ToBeWritten之车联网安全中常见的TOP 10漏洞
也许每个人出生的时候都以为这世界都是为他一个人而存在的,当他发现自己错的时候,他便开始长大 少走了弯路,也就错过了风景,无论如何,感谢经历 转移发布平台通知:将不再在CSDN博客发布新文章,敬…...
软考-密码学概述
本文为作者学习文章,按作者习惯写成,如有错误或需要追加内容请留言(不喜勿喷) 本文为追加文章,后期慢慢追加 by 2023年10月 密码学基本概念 密码学的主要目的是保持明文的秘密以防止攻击者获知,而密码分…...

windows 2003、2008远程直接关闭远程后设置自动注销会话
1、2003系统: 按开始—运行—输入“tscc.msc”,打开“终端服务配置”。 单击左边窗口的“连接”项,右边窗口中右击“RDP-TCP”,选择“属性”。 单击“会话”项,勾选“替代用户设置”,在“结束已断开的会话”…...
iOS BUG UIView转UIImage模糊失真
iOS BUG UIView转UIImage模糊失真 ##UIView转成Image - (UIImage *)capture {UIGraphicsBeginImageContextWithOptions(self.bounds.size, YES, 0.0);[self.layer renderInContext:UIGraphicsGetCurrentContext()];UIImage *img UIGraphicsGetImageFromCurrentImageContext(…...

如何在10分钟内让Android应用大小减少 60%?
一个APP的包之所以大,主要包括一下文件 代码libso本地库资源文件(图片,音频,字体等) 瘦身就主要瘦这些。 一、打包的時候刪除不用的代码 buildTypes {debug {...shrinkResources true // 是否去除无效的资源文件(如…...
网络代理技术:保障隐私与增强安全
在当今数字化的世界中,网络代理技术的重要性日益凸显。无论您是普通用户还是网络工程师,了解如何使用代理技术来保护隐私和增强网络安全都是至关重要的。本文将深入探讨Socks5代理、IP代理以及它们在网络安全和隐私保护中的关键作用。 1. Socks5代理&am…...

数据结构 | (二) List
什么是 List 在集合框架中, List 是一个接口,继承自 Collection 。 Collection 也是一个接口 ,该接口中规范了后序容器中常用的一些方法,具体如下所示: Iterable 也是一个接口,表示实现该接口的类是可以逐个…...

[NewStarCTF 2023 公开赛道] week1 Crypto
brainfuck 题目描述: [>>>>>>>>>>>>>>>><<<<<<<<<<<<<<<<-]>>>>>>>.>----.<-----.>-----.>-----.<<<-.>>..…...
C语言中文网 - Shell脚本 - 0
教程目录如下: 第1章 Shell基础(开胃菜) 1. Shell是什么?1分钟理解Shell的概念! 2. Shell是运维人员必须掌握的技能 3. 常用的Shell有哪些? 4. 进入Shell的两种方式 5. Linux Shell命令的基本格式 6.…...
应用升级/灾备测试时使用guarantee 闪回点迅速回退
1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间, 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点,不需要开启数据库闪回。…...

(转)什么是DockerCompose?它有什么作用?
一、什么是DockerCompose? DockerCompose可以基于Compose文件帮我们快速的部署分布式应用,而无需手动一个个创建和运行容器。 Compose文件是一个文本文件,通过指令定义集群中的每个容器如何运行。 DockerCompose就是把DockerFile转换成指令去运行。 …...

关键领域软件测试的突围之路:如何破解安全与效率的平衡难题
在数字化浪潮席卷全球的今天,软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件,这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下,实现高效测试与快速迭代?这一命题正考验着…...

sipsak:SIP瑞士军刀!全参数详细教程!Kali Linux教程!
简介 sipsak 是一个面向会话初始协议 (SIP) 应用程序开发人员和管理员的小型命令行工具。它可以用于对 SIP 应用程序和设备进行一些简单的测试。 sipsak 是一款 SIP 压力和诊断实用程序。它通过 sip-uri 向服务器发送 SIP 请求,并检查收到的响应。它以以下模式之一…...

初探Service服务发现机制
1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能:服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源…...
快刀集(1): 一刀斩断视频片头广告
一刀流:用一个简单脚本,秒杀视频片头广告,还你清爽观影体验。 1. 引子 作为一个爱生活、爱学习、爱收藏高清资源的老码农,平时写代码之余看看电影、补补片,是再正常不过的事。 电影嘛,要沉浸,…...

pikachu靶场通关笔记19 SQL注入02-字符型注入(GET)
目录 一、SQL注入 二、字符型SQL注入 三、字符型注入与数字型注入 四、源码分析 五、渗透实战 1、渗透准备 2、SQL注入探测 (1)输入单引号 (2)万能注入语句 3、获取回显列orderby 4、获取数据库名database 5、获取表名…...

高考志愿填报管理系统---开发介绍
高考志愿填报管理系统是一款专为教育机构、学校和教师设计的学生信息管理和志愿填报辅助平台。系统基于Django框架开发,采用现代化的Web技术,为教育工作者提供高效、安全、便捷的学生管理解决方案。 ## 📋 系统概述 ### 🎯 系统定…...
React从基础入门到高级实战:React 实战项目 - 项目五:微前端与模块化架构
React 实战项目:微前端与模块化架构 欢迎来到 React 开发教程专栏 的第 30 篇!在前 29 篇文章中,我们从 React 的基础概念逐步深入到高级技巧,涵盖了组件设计、状态管理、路由配置、性能优化和企业级应用等核心内容。这一次&…...
[特殊字符] 手撸 Redis 互斥锁那些坑
📖 手撸 Redis 互斥锁那些坑 最近搞业务遇到高并发下同一个 key 的互斥操作,想实现分布式环境下的互斥锁。于是私下顺手手撸了个基于 Redis 的简单互斥锁,也顺便跟 Redisson 的 RLock 机制对比了下,记录一波,别踩我踩过…...