`
izuoyan
  • 浏览: 9295938 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

WinEyes的重新实现--windows系统及其消息机制

阅读更多

windows和x系统是相似的,然而它既不以进程为根本,又不以线程为根本,而是以窗口为根本的,由于它的过程的每一个环节都是在本机进行的,所以它必然需要在进程,线程以及窗口之间进行更进一步的细分,毕竟在没有虚拟机硬件的支持下,一台机器的最小元素就是线程(在多处理情况下)。具体过程是,当有鼠标键盘或者触摸屏之类的东西有输入事件时,windows系统会将这个事件排入到当前进程的当前线程的消息队列中,注意是基于线程的,然后由当前进程应用程序通过GetMessge这个API依次取出一个消息,然后再次压入windows系统,而后windows系统将这个消息分发到当前的窗口,注意这个过程中,将消息分派到当前窗口是通过线程来进行的,第一个过程是系统将事件排入当前线程,第二个过程是系统将一个消息排入当前线程的窗口,因为一个线程可能不仅仅包括一个窗口,windows系统的一切可操作元素都是通过窗口来组织的(witget),就连标题栏也是一个窗口。
这就可以看出,windows的GUI和x系统的GUI的不同之处,由于x系统中,是通过socket这种进程间通信的方式来进行通信的,所以输入设备和显示这种事天生就是接触耦合的,况且,显示图形的机制是用户空间完成,输入设备通过设备文件(/dev/input/mice...)被读到用户空间,然后x server进程通过x网络协议分发这些事件到各个x client。x系统的进程间通信天生就不假设是同一台机器的,因此就不能用线程这种轻量级的机制来进一步裁剪机制--毕竟线程不能在不同机器间被认同--它们不能共享进程地址空间,所以x系统的最小单元就是进程。在windows系统中,由于事件和显示都是同一台机器完成的,所以尽可以充分使用操作系统内部的机制,譬如线程来更加高效的完成,因此windows操作系统中就充分使用了线程消息队列,然后在线程环境中将各个消息分派到当前窗口,这是因为这一点,GetMessage之后不能手工调用窗口过程函数,而要通过DispatchMessage函数来分发同一线程的消息到可能有多个的同一线程的不同窗口的之一。
明白了windows的消息处理原理,很容易写出一个windows版本的xeyes程序,然而在看既有的wineyes程序源代码的时候,发现它也是通过timer实现的,这个令人很不爽,正如在linux中遇到的那样!因此有必要改造之,既然我们的linux改造版的xeyes使用了和windows类似的事件循环机制,那么我们的windows也可以和linux更加靠拢,那就是不调用DispatchMessage函数,而直接调用窗口处理函数,前提是消息确实是发送给我们这个窗口的--毕竟消息是从线程的消息队列中取出的而消息的目的地却是基于窗口的--一个线程不止一个窗口!整个思路是这样的,我们一直都以为windows的处理机制和x window有着类似之处,只不过windows由于x server和x client处于一个操作系统空间,因此它们可以充分使用操作系统的特性,比如线程,比如无消息时调度等特性。既然它们的本质是一样的,那么它们的实现也可以相通,既然x系统可以使用xlib来实现一个消息循环,那么windows也是可以的,只要我们排除掉它使用的操作系统特性即可,因此,我们加一个限制条件:
while(GetMessage(...)) {
TranslateMessage(...);
if (msg.hwnd == hWnd) {
直接调用窗口函数
} else {
DispatchMessage(...);
}
}
可见,只要直接调用窗口函数就可以了,而没有必要让操作系统dispatch消息了。然而除了因为一个线程可能对应多个窗口导致的消息不一定是给我们的这个原因外,还有什么原因导致必须调用dispatch呢?这和windows基于线程调度是有一定关系的,这与其说是一种完美的设计不如说是一种败笔,之所以这么设计为的就是操作系统更好的监控窗口函数的处理情况,如果直接调用窗口而不是调用dispatch,那么操作系统完全不知道你的窗口函数什么时候返回的,如果你的窗口函数正在绘图,此时将该线程调度出去是不合适的,除非它占用了太久的cpu或者有更要紧的事要做,因此windows操作系统最好能设法监控到窗口函数什么时候返回,以决定调度时机。这个机制无论如何都太复杂了,因此就简单性原则来说就是败笔,x系统就不需要这么做,因为它基于task_struct调度,简单,高效,不管task_struct到底是什么,因此linux也根本不觉得将正在绘图的task_struct调度出去有什么不合适的,在windows中,GUI有特权,在linux中GUI只是一个X进程,它没有特权,一般般!
下面就是wineyes的无timer版本,也不知道为什么,XXeyes怎么都使用timer来做,搞不懂:
...
#pragma comment(linker, "/subsystem:\"console\" /entry:\"WinMainCRTStartup\"") //便于调试,因为有console窗口
...
void setClippingRegion(HWND hWnd)
{
...//完全是wineyes的
}
void WinEyesUpdate(HWND hWnd, int ForceRedrawEyes)
{
...//完全是wineyes的
}
void WinEyesPaint(HWND hWnd)
{
...//完全是wineyes的
}
...
LRESULT CALLBACK PASCAL WinEyesWndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
...//完全是wineyes的,但是去掉了WM_TIMER,因为这是无timer版的
}
BOOL WinEyesInit(HINSTANCE hInstance)
{
...//完全是wineyes的
}
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{

while( GetMessage( &msg, NULL, (int)NULL, (int)NULL) ) { //和xlib类似的消息(事件)循环
TranslateMessage(&msg);
if (hWnd == msg.hwnd) { //如果消息是发给本窗口的,直接调用
WinEyesWndProc(msg.hwnd, msg.message, msg.wParam, msg.lParam);
}
else //注意,这是在从线程消息队列取消息,它可能不属于我们这个窗口,如果这样就调用dispatch,将之交由操作系统来处理,其实是操作系统负责将之路由到msg.hwnd
DispatchMessage(&msg);
}
...
}

分享到:
评论

相关推荐

    WinEyes-开源

    WinEyes是xeyes实用程序(由Keith Packard和MIT X Consortium重写)的,用于Win32,Pocket PC,ReactOS和Wine。 WinEyes 的目的是尽可能忠实于 xeyes 的原始外观。

    S变换+Sockwell R G , Mansinha L , Lowe R P . Localization of the complex spectrum: the S transformJ

    s变换用的高斯窗函数( 高斯窗是指数窗的一种,它也无负的旁瓣,而且没有旁瓣波动,因而不回引起计算谱中假的极大值或极小值,而且高斯窗频率窗函数的主瓣比指数窗的主瓣窄,分辨率比指数窗有所提高。

    2021科大讯飞车辆贷违预测大赛冠军源码+全部资料.zip

    2021科大讯飞车辆贷违预测大赛冠军源码+全部资料.zip [资源说明] 1、该项目是团队成员近期最新开发,代码完整,资料齐全,含设计文档等 2、上传的项目源码经过严格测试,功能完善且能正常运行,请放心下载使用! 3、本项目适合计算机相关专业(人工智能、通信工程、自动化、电子信息、物联网等)的高校学生、教师、科研工作者、行业从业者下载使用,可借鉴学习,也可直接作为毕业设计、课程设计、作业、项目初期立项演示等,也适合小白学习进阶,遇到问题不懂就问,欢迎交流。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 5、不懂配置和运行,可远程教学 欢迎下载,学习使用!

    AI图像处理工具包-一键抠图、背景切换、旧照片修复、人像漫画化、视频卡通化(Python+OpenCV+Dlib+TensorFlow).zip

    AI图像处理工具包-一键抠图、背景切换、旧照片修复、人像漫画化、视频卡通化(Python+OpenCV+Dlib+TensorFlow).zip [资源说明] 1、该项目是团队成员近期最新开发,代码完整,资料齐全,含设计文档等 2、上传的项目源码经过严格测试,功能完善且能正常运行,请放心下载使用! 3、本项目适合计算机相关专业(人工智能、通信工程、自动化、电子信息、物联网等)的高校学生、教师、科研工作者、行业从业者下载使用,可借鉴学习,也可直接作为毕业设计、课程设计、作业、项目初期立项演示等,也适合小白学习进阶,遇到问题不懂就问,欢迎交流。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 5、不懂配置和运行,可远程教学 欢迎下载,学习使用!

    基于java+springboot+vue+mysql的远程教育网站设计与实现.docx

    基于java+springboot+vue+mysql的远程教育网站设计与实现.docx

    springboot005学生心理咨询评估系统(源码+数据库+论文+PPT+包调试+一对一指导)

    毕业设计资料,计算机毕业设计,源码,毕业论文,毕业答辩,答辩PPT,Java毕业设计,php毕业设计,ASP.NET毕业设计,毕业指导,计算机作业,php作业,java作业,ASP.NET作业,编程作业,管理系统,网站,app,毕业设计学习,Java学习,php学习,ASP.NET学习,java课程,php课程,ASP.NET课程,答辩技巧,SQLSERVER数据库,Mysql数据库,jdbc,SSM框架,SpringBoot框架,Html5,小程序

    蓝牙串口助手,可以连接HC-05等蓝牙模块,实现单片机设备与手机通讯,安卓手机,蓝牙调试助手,具有按键功能!

    蓝牙串口助手,可以连接HC-05等蓝牙模块,实现单片机设备与手机通讯,安卓手机,蓝牙调试助手,具有按键功能!

    TriLib-2-Model-Loading-Package-2.3.7.unitypackage

    TriLib 2 是一个跨平台的运行时 3D 模型导入器

    “人力资源+大数据+薪酬报告+涨薪调薪”

    人力资源+大数据+薪酬报告+涨薪调薪,在学习、工作生活中,越来越多的事务都会使用到报告,通常情况下,报告的内容含量大、篇幅较长。那么什么样的薪酬报告才是有效的呢?以下是小编精心整理的调薪申请报告,欢迎大家分享。相信老板看到这样的报告,一定会考虑涨薪的哦。

Global site tag (gtag.js) - Google Analytics