基于截图和模拟点击的自动化压测工具开发(MFC)
1.背景
想对一个MFC程序做自动压测功能,根据判断程序界面某块区域是否达到预定状态,来自动执行鼠标点击或者键盘输入的操作,以解决测试人员需要重复手动压测问题。
1.涉及的技术
串口控制,基于MFC橡皮筋类(CRectTracker)做一个简单的截图对话框,GDI,模拟鼠标点击(mouse_event),模拟键盘输入(keybd_event),MD5等
2.串口控制继电器上下电
如何你要压测的程序需要通过控制USB上下电,可用串口控制继电器来达到此目的。
- 以下是串口初始化的代码
bool CComTest::InitialCom(int iComID, int iComPort, DWORD iBaudRate)
{if (iComPort < 1 || iComPort > 255){return FALSE;}if (INVALID_HANDLE_VALUE != m_hCom){CloseCom(iComID);}if (INVALID_HANDLE_VALUE != m_hCom){CloseHandle(m_hCom);}m_iComPort = iComPort; //串口通信端口m_iBaudRate = iBaudRate; //串口通信速率DCB Dcb;CString str;COMMTIMEOUTS TimeOut;int Data = 8;int Stop = 0;int Parity = 0;str.Format(_T("COM%d"), iComPort);m_hCom = CreateFile(str, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, NULL, NULL);if (m_hCom == INVALID_HANDLE_VALUE){return FALSE;}GetCommState(m_hCom, &Dcb);Dcb.BaudRate = iBaudRate;Dcb.ByteSize = Data;Dcb.StopBits = Stop;Dcb.Parity = Parity;if (!SetCommState(m_hCom, &Dcb)){CloseHandle(m_hCom);m_hCom[iComID] = INVALID_HANDLE_VALUE;return FALSE;}memset(&TimeOut, 0, sizeof(TimeOut));TimeOut.ReadIntervalTimeout = MAXDWORD;SetCommTimeouts(m_hCom[iComID], &TimeOut);SetupComm(m_hCom, 1024, 1024);PurgeComm(m_hCom, PURGE_TXABORT | PURGE_RXABORT | PURGE_TXCLEAR | PURGE_RXCLEAR);return TRUE;
}
- 读取串口:
DWORD CComTest::ReadCom(int ComNo, BYTE *pBuff, int nCount)
{if (m_hCom == INVALID_HANDLE_VALUE){return 0;}DWORD read = 0;ReadFile(m_hCom, pBuff, nCount, &read, NULL);return read;
}
写串口
BOOL CComTest::WriteCom(int ComNo, BYTE *pBuff, int nCount)
{if (m_hCom == INVALID_HANDLE_VALUE){return FALSE;}DWORD written = 0;BOOL ret = WriteFile(m_hCom, pBuff, nCount, &written, NULL);return ret;
}
发串口命令示例
BYTE pData[6] = { 0x01, 0x02, 0x03, 0x04, 0x05, 0x06 };WriteCom(m_iComID, pData, 6);
继电器控制命令需参考具体供应商提供的使用文档。
3.程序界面状态监控(截图对话框的实现)
想要通过检测某块区域是否达到预设的状态来自动执行被压测程序的下一步动作,可在压测前将被压测程序执行到预定状态,然后截取该区域的位图并保存,在压测时不停的检测该区域位图,和前面保存的位图做对比(MD5),如果MD5一致则执行模拟点击或模拟输入动作。
- 截图对话框的OnInitDialog函数代码如下
BOOL CCaptureDlg::OnInitDialog()
{CDialog::OnInitDialog();copyScreenToBitmap(m_ScreenBmp); //将屏幕内容拷贝到Bitmap类型的对象中//获取屏幕当前分辨率的宽度和高度(以像素为单位,传递参数不同X,Y)int screenWidth = GetSystemMetrics(SM_CXSCREEN);int screenHeight = GetSystemMetrics(SM_CYSCREEN);//调用MoveWindow或者SetWindowPos将当前的窗口设置成与屏幕大小相同//使用两个,软件写好后要使用SetWindowPos,但使用SetWindowPos设置成顶层窗口就不能调试了,使用MoveWindow进行调试MoveWindow(-3, -3, screenWidth + 6, screenHeight + 6); //比屏幕膜大3个像素不然白边出现,好看一些//SetWindowPos(&wndTopMost, -3, -3, screenWidth + 6, screenHeight + 6, SWP_SHOWWINDOW);//橡皮筋类的操作m_rectTracker.m_nStyle = CRectTracker::resizeOutside | CRectTracker::dottedLine; //矩形框虚线m_rectTracker.m_rect.SetRect(0, 0, 0, 0); //初始化矩形大小return TRUE; // return TRUE unless you set the focus to a control
}
m_ScreenBmp会保存截图对话框显示前的屏幕实时位图。
- 截图对话框背景显示原来屏幕的背景:
BOOL CCaptureDlg::OnEraseBkgnd(CDC* pDC)
{//将bitmap对象作为背景画到对话框上//创建内存DCCDC memDC;memDC.CreateCompatibleDC(pDC); //使内存DC与pDC兼容memDC.SelectObject(&m_ScreenBmp); //选入设备环境//将内容从内存DC拷贝到pDC中(本模态对话框窗口的DC)CRect rect;GetClientRect(&rect); //获取对话框大小,在初始化时设置了pDC->BitBlt(0, 0, rect.Width(), rect.Height(), &memDC, 0, 0, SRCCOPY); //将内容从内存DC拷贝到pDC中memDC.DeleteDC(); //释放return TRUE; //直接在此返回,不进行下一步操作,否则就画不上了return CDialog::OnEraseBkgnd(pDC);
}
- 鼠标消息处理,使橡皮筋对象矩形可操作:
void CCaptureDlg::OnLButtonDown(UINT nFlags, CPoint point)
{//如果点击绘制橡皮筋区域的外部,重新构建一个可拖拽的区域if (m_rectTracker.HitTest(point) == CRectTracker::hitNothing){m_rectTracker.TrackRubberBand(this, point, TRUE);}else{//点击在了区域内部,允许用户大小调整进行区域描画m_rectTracker.Track(this, point, TRUE);m_rectTracker.m_rect.NormalizeRect(); //NormalizeRect可以进行左右上下值调整,从右下向左上框柱}Invalidate(TRUE); //更新,使WM_PAINT描画消息触发CDialog::OnLButtonDown(nFlags, point);
}
- 修改橡皮筋框类光标的形态:
BOOL CCaptureDlg::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)
{//如果传递的是本窗口,SetCursor成功了if (pWnd == this && m_rectTracker.SetCursor(this, nHitTest)){SetCursor(LoadCursor(NULL, IDC_CROSS));return TRUE;}else{return CDialog::OnSetCursor(pWnd, nHitTest, message);}
}
- 在OnPaint中实时显示橡皮筋框
void CCaptureDlg::OnPaint()
{//CPaintDC只适合OnPaint里面,所以使用GetDC来获取DC,进行描绘CDC *pDC = GetDC();m_rectTracker.Draw(pDC);ReleaseDC(pDC); //get与release成对使用
}
- 双击时将橡皮筋对话框坐标和大小保存,并此区域内的对话框未显示前的屏幕位图保存为图片
void CCaptureDlg::OnLButtonDblClk(UINT nFlags, CPoint point)
{// 保存双击时坐标m_startPointX = point.x;m_startPointY = point.y;//如果双击在了矩形区域的内部就进行保存工作if (m_rectTracker.HitTest(point) != CRectTracker::hitMiddle){ MessageBox(_T("截图失败,请重新截图!"));return;}CDC *pDC = GetDC();CDC memDC;memDC.CreateCompatibleDC(pDC);memDC.SelectObject(&m_ScreenBmp);CRect rect;rect = m_rectTracker.m_rect;CBitmap mBmp, *pOldBmp = NULL;mBmp.CreateCompatibleBitmap(pDC, rect.Width(), rect.Height());m_rectInfo.m_Top1 = rect.top;m_rectInfo.m_Bottom1 = rect.bottom;m_rectInfo.m_Left1 = rect.left;m_rectInfo.m_Right1 = rect.right;CDC dstDC;dstDC.CreateCompatibleDC(pDC);pOldBmp = dstDC.SelectObject(&mBmp);//内容拷贝到目标,目标肯定比屏幕小,所以坐标0,0,宽高用户选择的,rect.left, rect.top是源缓冲区的坐标dstDC.BitBlt(0, 0, rect.Width(), rect.Height(), &memDC, rect.left, rect.top, SRCCOPY);CImage Img;Img.Attach(mBmp); //关联CString strModuleCapturePath;CString strCapturePath = "你想要保存的路径\\capture1.jpg";Img.Save(strCapturePath ); //保存地址mBmp.DeleteObject();memDC.DeleteDC();dstDC.DeleteDC();ReleaseDC(pDC);//保存完了后,说明本次截图操作完成了,要把当前显示的模态对话框,全屏的对话框关闭CDialog::OnCancel();
}
- 以上步骤可实现压测前某块区域的位图、区域和双击坐标的保存,以下函数可截取屏幕某个区域的位图并保存到本地:
void CCaptureDlg::Screen(CRect cRect, int iNumber)
{CRect rect = cRect;CDC *pDC;//屏幕DCpDC = CDC::FromHandle(::GetDC(NULL));//获取当前整个屏幕DCint BitPerPixel = pDC->GetDeviceCaps(BITSPIXEL);//获得颜色模式CDC memDC;//内存DCmemDC.CreateCompatibleDC(pDC);CBitmap memBitmap, *oldmemBitmap;//建立和屏幕兼容的bitmapmemBitmap.CreateCompatibleBitmap(pDC, rect.Width(), rect.Height());oldmemBitmap = memDC.SelectObject(&memBitmap);//将memBitmap选入内存DCmemDC.BitBlt(0, 0, rect.Width(), rect.Height(), pDC, rect.left, rect.top, SRCCOPY);//备注CImage Img;Img.Attach(memBitmap); //关联CString strCapturePath = _T("你想要保存的路径\\capture1.jpg");memDC.SelectObject(oldmemBitmap);memDC.DeleteDC(); ::ReleaseDC(NULL, pDC->m_hDC);
}
4.模拟鼠标点击
可通过以下代码来实现:
SetCursorPos(m_strPointX, m_strPointY);
mouse_event(MOUSEEVENTF_LEFTDOWN, m_strPointX, m_strPointY, 0, 0);
mouse_event(MOUSEEVENTF_LEFTUP, m_strPointX, m_strPointY, 0, 0);
5.模拟键盘输入
比如想在某个编辑框中自动输入字符串,可使用前面叙述的截图对话框截取编辑框区域,并通过模拟鼠标点击将光标焦点移动到编辑框内(SetCursorPos的坐标使用双击截图对话框时保存的坐标)
// 截取编辑框时,双击的坐标
SetCursorPos(m_strPointX, m_strPointY);
mouse_event(MOUSEEVENTF_LEFTDOWN, m_strPointX, m_strPointY, 0, 0);
mouse_event(MOUSEEVENTF_LEFTUP, m_strPointX, m_strPointY, 0, 0);
char szImput[10] = "123456789";
keybd_event(szImput[0], 0, 0, 0);
...
keybd_event(szImput[9], 0, 0, 0
相关文章:
基于截图和模拟点击的自动化压测工具开发(MFC)
1.背景 想对一个MFC程序做自动压测功能,根据判断程序界面某块区域是否达到预定状态,来自动执行鼠标点击或者键盘输入的操作,以解决测试人员需要重复手动压测问题。 1.涉及的技术 串口控制,基于MFC橡皮筋类(CRectTracker)做一个…...
力扣每日一题 6/10
881.救生艇[中等] 题目: 给定数组 people 。people[i]表示第 i 个人的体重 ,船的数量不限,每艘船可以承载的最大重量为 limit。 每艘船最多可同时载两人,但条件是这些人的重量之和最多为 limit。 返回 承载所有人所需的最小船…...
[知识点] 内存顺序属性的用途和行为
C标准库中定义了以下几种内存顺序属性: std::memory_order_relaxedstd::memory_order_consumestd::memory_order_acquirestd::memory_order_releasestd::memory_order_acq_relstd::memory_order_seq_cst 1. std::memory_order_relaxed 定义:不提供同步…...
JAVA Mongodb 深入学习(二)索引的创建和优化
一、常用索引类型 1、单个索引 单个索引的创建 db.你的表名.createIndex({"你的字段名":1}) 单个索引的创建且是唯一索引 db.你的表名.createIndex({"你的字段名":1}),{ unique: true }) 2、复合索引 将多个过滤的字段,做成索引,…...
转让北京劳务分包地基基础施工资质条件和流程
地基基础资质转让流程是怎样的?对于企业来说,资质证书不仅是实力的证明,更是获得工程承包的前提。而在有了资质证书后,企业才可以安心的准备工程投标,进而在工程竣工后获得收益。而对于从事地基基础工程施工的企业,需…...
Python基础——字符串
一、Python的字符串简介 Python中的字符串是一种计算机程序中常用的数据类型【可将字符串看作是一个由字母、数字、符号组成的序列容器】,字符串可以用来表示文本数据。 通常使用一对英文的单引号()或者双引号(")…...
AP的数据库性能到底重要吗?
先说结论:没那么重要。甚至可能不重要。 我用我的经历和分析给大家说说。诸位看看如何。 不重要的观点是不是不能接受? 因为这些是站在我们角度觉得的。而实际上使用者(业务或者用户),真的不太在乎我们所在乎的。 …...
Vue3【二】 VSCode需要安装的Vue语法插件
VSCode需要安装的 适配Vue3的插件 Vue-Official插件安装...
设置路径别名
一、描述 如果想要给路径设置为别名,就是常见的有些项目前面的引入文件通过开头的,也就是替换了一些固定的文件路径,怎么配置。 二、配置 import { defineConfig } from vite import react from vitejs/plugin-react import path from path…...
人事信息管理系统(Java+MySQL)
一、项目背景 在现代企业中,管理大量员工的工作信息、薪资、请假、离职等事务是一项非常繁琐和复杂的任务。传统的手工管理方式不仅效率低下,而且容易出错。为了提高人事管理的效率,减少人工操作带来的错误,企业迫切需要一个高效…...
Python 中生成器与普通函数的区别
在Python中,生成器和普通函数有一些区别。 生成器使用 yield 语句从函数中返回一个值,而不是使用 return 语句。当生成器函数被调用时,它会返回一个迭代器对象,而非立即执行函数体内的代码。 生成器函数可以通过多次调用 yield 语…...
最小栈、栈的弹出(C++)
1.最小栈 思路分析: 代码: class MinStack { public:MinStack() {}void push(int val) {st.push(val);//两种情况需要更新最小值//1.最小栈为空(就是存最小值的那个栈)//2.插入的值小于或等于最小栈的栈顶元素if(minstack.empty()||minstack.top()>…...
20240607每日通信--------VUE3前端引入scoket-io,后端引入Netty-SocketIO,我成功了,希望一起交流沟通
无语 前置: VUE3 前端集成scoket-io socket.io-client Sringboot 3.0JDK17集成Netty-SocketIO Netty-SocketIO 失败原因一: 前期决定要写demo时候,单独了解了,后端引入Netty-SocketIO注意事项,详见我先头写的博客 前…...
Tomcat源码解析(八):一个请求的执行流程(附Tomcat整体总结)
Tomcat源码系列文章 Tomcat源码解析(一):Tomcat整体架构 Tomcat源码解析(二):Bootstrap和Catalina Tomcat源码解析(三):LifeCycle生命周期管理 Tomcat源码解析(四):StandardServer和StandardService Tomcat源码解析(五)&…...
python使用gdb进行堆栈查看与调试
以ubuntu示例,先安装gdb与python-dbg,dbg按照python版本安装 apt install -y gdb python3.10-dbg 使用top查看python进程,使用gdb操作python进程 gdb python3 6618 加载环境 source /usr/share/gdb/auto-load/usr/bin/python3.10-gdb.py…...
【DevOps】路由与路由器详细介绍:原理、功能、类型及应用场景
目录 一、路由详细介绍 1、什么是路由? 2、路由的基本原理 3、 路由协议 静态路由 动态路由 4、 路由表 5、 路由算法 6、路由的优缺点 优点 缺点 7、 路由应用场景 二、路由器详细介绍 1、什么是路由器? 2、 路由器的工作原理 3、路由器…...
【WP|9】深入解析WordPress [add_shortcode]函数
add_shortcode 是 WordPress 中一个非常强大的函数,用于创建自定义的短代码(shortcodes)。短代码是一种简洁的方式,允许用户在内容中插入动态的、可重用的功能。通过 add_shortcode,开发者可以定义自己的短代码&#x…...
Qt QStackedWidget类详细分析
一.定义 QStackedWidget类是一个容器控件,它提供了一个堆叠的页面布局方式,每个页面可以包含一个子部件。在QStackedWidget中,只有当前活动的页面是可见的,其他页面会被隐藏起来。 QStackedWidget类的常用方法包括: a…...
Java数据结构与算法(leetcode热题881. 救生艇)
前言 救生艇属于贪心算法,解题之前条件一定要归纳好。题目中存在3个要求: 1.一艘船最多坐2人 2.船数要求最小 3.每艘船重量小于limit 意味着体重较轻的两人可以同乘一艘救生艇。 . - 力扣(LeetCode) 实现原理 1.重量大的有…...
react+wijmo所遇问题
1.官网地址:https://demo.mescius/wijmo/demos/Grid/Overview/react 别进中文地址,注意后缀mescius有没有.cn有的话删掉,那个没有触发方法和各类API,组件也不全 2.中文地址:(不太好用)&#x…...
手撕设计模式——克隆对象之原型模式
1.业务需求 大家好,我是菠菜啊,前俩天有点忙,今天继续更新了。今天给大家介绍克隆对象——原型模式。老规矩,在介绍这期之前,我们先来看看这样的需求:《西游记》中每次孙悟空拔出一撮猴毛吹一下&#x…...
LangChain基础知识入门
LangChain的介绍和入门 1 什么是LangChain LangChain由 Harrison Chase 创建于2022年10月,它是围绕LLMs(大语言模型)建立的一个框架,LLMs使用机器学习算法和海量数据来分析和理解自然语言,GPT3.5、GPT4是LLMs最先进的代…...
Objective-C的初始化方法中,应该如何读写属性
除非有明确的原因需要使用setter, getter, 否则总是应该直接访问, 也就是直接使用实例变量(也称为 iVar)来读写数据 理由: 避免子类覆盖setter方法的影响:若在初始化方法中使用setter方法, 使用此方法实例化子类, 可能会调用子类…...
基于Python+Flask框架实现的新冠疫情可视化的设计与实现
基于PythonFlask框架实现的新冠疫情可视化的设计与实现 “Design and Implementation of COVID-19 Visualization using Python Flask Framework” 完整下载链接:基于PythonFlask框架实现的新冠疫情可视化的设计与实现 文章目录 基于PythonFlask框架实现的新冠疫情可视化的设…...
大学生如何学习C语言编程?
设计语言》(K&R)和《C Primer Plus》。 安装开发环境:安装一个C语言编译器,如GCC,以及一个集成开发环境(IDE),比如Code::Blocks或Visual Studio。 学习语法:熟悉C语…...
python小tips
函数: 格式: def 函数的名字():函数体例如:def playgame():print("I am playing!")函数调用: playgame()调用的方法: 函数名() 函数的定义只是定义函数,调用了才会有结果 函数的参…...
分布式版本控制工具软件——Git概述
目录 一、Git概述1.为什么要学习Git?(1)SCM概念(2)SCM实现 2.什么是版本控制?(1)版本控制软件的基础功能(2)集中式版本控制(3)分布式版…...
【一百零八】【算法分析与设计】P1908 逆序对,P1637 三元上升子序列,树状数组区间和应用
P1908 逆序对 逆序对 题目描述 猫猫 TOM 和小老鼠 JERRY 最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计。 最近,TOM 老猫查阅到一个人类称之为“逆序对”的东西,这东西…...
【RK3568】制作Android11开机动画
Android 开机 logo 分为两种:静态显示和动态显示。静态显示就是循环显示一张图片;动态显示就是以特定帧率顺序显示多张图片 1.准备 android logo 图片 Android logo最好是png格式的,因为同一张图片的情况下,png 格式的比 jpg和b…...
chrony内网同步服务器时间
当前需要在10.26.24.62和10.26.24.61两个服务器上设置chrony同步时间,其中10.26.24.62为NTP时间服务器,10.26.24.61去10.26.24.62同步时间 检查Chrony配置文件: 确认10.26.24.62(NTP服务器)的配置文件 /etc/chrony/c…...
学生做网站怎么收费/郑州seo方案
话题一:异或先看一段代码吧:这段代码是用来实现a,b交换的。异或是位运算符,0^1为1,其他情况都是0,这是一个加法不进位的功能!这有什么用呢?其实通过上面交换的例子,我们已经明白A^B^…...
做游戏直播什么游戏视频网站好/免费的网站域名查询
一、使用top命令查看占用高资源的java项目的进程ID(pid): top 二、查看该进程中的线程所占用资源的情况:top -Hp pid 三、查看该线程对应的16进制:printf %x 11129 打印并保存该进程中堆栈的使用信息日志:jstack -l 11095 >> jstack.lo…...
做什么网站能吸引流量/长沙靠谱关键词优化服务
建议的采用顺序是List中泛型顺序依次为T、?、Object (1)、List是确定的某一个类型 List表示的是List集合中的元素都为T类型,具体类型在运行期决定;List<?>表示的是任意类型,与List类似,而List则表示List集合中的…...
做外贸网站要花多少钱/seo软件推广哪个好
ios打包和调试 Mac系统安装XCode方便调试 在App Store下载并安装Xcode。注意,xcode版本低的话,也会影响打包的。推荐使用XCode9. 如果你的OS版本不满足安装xcode的版本要求,你可以寻找对应版本的xcode下载链接,然后在safari打开…...
做个小程序多少钱/企业网站如何优化
目录一、linux安装jdk二、linux安装tomcat三、linux安装idea2020四、linux安装mysql一、linux安装jdk 7.在 /etc/profile 文件中配置环境变量: 10.让环境变量 的文件生效: source /etc/profile 然后在控制台输出环境变量: echo $PATH 如下&…...
做徽商要做网站吗/域名大全查询
1、视频降低质量,减小体积: ffmpeg -i aaa.mp4 -strict -2 -qscale 20 -y outfile.mp4转载于:https://www.cnblogs.com/lanqie/p/8302530.html...