1       故障现象

故障现象1:能够收到SendMessage()发出的消息,但收不到PostMessage()发出的消息。

故障现象2:能够收到PostMessage()发出的消息,但收不到SendMessage()发出的消息。

2       两个比喻

从发送端来说:

SendMessage()好比电信系统提供通话服务。

如果对方占线,立即返回。

如果对方不占线,就保持呼叫→接收端接听→通话→挂断。在挂断之前,发送端处于阻塞状态,不可以干别的事。

PostMessage()好比电子邮件系统提供电子邮件服务。

发送端把邮件交给电子邮件系统,然后立即返回,就可以干别的事了。

电子邮件系统将邮件放在队列缓存区中,缓慢地将邮件发送到接收端的邮箱里。

从接收端来说(!!! 关键 关键 关键 !!!)

要接收SendMessage()的消息,必须用手机接听,到邮箱里无法接收此类消息。

要接收PostMessage()的消息,必须到邮箱里接收,用手机无法接听此类消息。

1       技术原理

在本文中:

把SendMessage()发送的消息成为立即消息。

把PostMessage()发送的消息成为延迟消息。

 

在计算机系统里,立即消息和延迟消息分别使用完全不同的途径发送消息,必须采用相匹配的、完全不同的接收方法才能收到消息。

1.1      立即消息的收发

为了接收并处理立即消息,接收端应用程序必须准备一张(消息,处理函数)映射表,如下表所示:

消息

处理函数

WM_USER + 1

OnWmUser_1

WM_USER + 2

OnWmUser_2

WM_USER + n

OnWmUser_n

立即消息的发送与接收的协作流程如下表:

发送端应用程序

操作系统

接收端应用程序

(消息处理总入口函数)

发送消息

阻塞发送端应用程序

调用接收端应用程序的消息处理总入口函数

在(消息,处理函数)映射表中搜索与消息相匹配的处理函数,立即调用处理函数。

向操作系统返回处理结果

向发送端应用程序返回处理结果

解除阻塞发送端应用程序

接收消息处理结果

继续其它事务

1.2      延迟消息的收发

延迟消息的发送与接收的协作流程如下表:

发送端应用程序

操作系统

接收端应用程序

发送消息

将消息放入接收端应用程序的消息队列中

继续其它事务

从消息队列中取出消息慢慢处理。

2       具体实现

以C++ Builder为例。

2.1      接收立即消息

要接收立即消息,也就是由SendMessage()发送的消息。

步骤1:声明并定义消息处理函数

步骤2:声明和定义(消息,处理函数)映射表

class TForm2 :   public TForm

{

// 声明WM_COPYDATA 消息的处理函数

void __fastcall OnCopyData(const   TMessage &msg);

// 受理立即消息 --- 由 SendMessage() 发送的消息

// 声明和定义(消息,处理函数)映射表

BEGIN_MESSAGE_MAP

VCL_MESSAGE_HANDLER(WM_COPYDATA, TMessage, OnCopyData);

END_MESSAGE_MAP(TForm)

};

2.2      接收延迟消息

要接收延迟消息,也就是由PostMessage()发送的消息。

放置一个TApplicationEvents控件,该控件只能接收延迟消息,无法接收立即消息。

编写TApplicationEvents.OnMessage事件处理函数。如下表:

void __fastcall   TForm2::AppEventsMessage(tagMSG &Msg, bool &Handled)

{

// 若收到延迟消息WM_USER_1

if (Msg.message == WM_USER_1)

{

// 此处编写消息处理代码

// 设立标志,消息已被处理

Handled   = true;

}

else {

// 设立标志,消息未被处理

Handled = false;

}

}

3       结论

SendMessage()和PostMessage()的共性:

都能发送消息。

SendMessage和PostMessage的差异:

采用完全不同的途径发送消息,也要采用完全不同的方法和途径接收消息。

若接收途径与发送途径不匹配,就无法接收消息。

通过SendMessage()发送的消息,发送端发送并必须等待接收端处理,接收端立即处理并向发送端反馈处理结果,无延迟。

通过PostMessage()发送的消息,发送端发送后立即返回,不等待接收端不处理,接收端缓慢地延迟处理。

 

SendMessage

PostMessage()

发送端

直接发给接收端

要求接收端立即处理

在接收端处理完毕之前处于阻塞状态,不可以干别的事

可以立即获悉接收端的处理结果

消息经由OS缓存到接收端的消息队列里

发送完毕立即返回,可以干别的事

接收端

立即处理收到的消息

缓慢地从队列里取出消息并处理。

4       其它

SendMessage和PostMessage的其它特性,网上都有,本文不再重复与累述。

迥异和诡异的SendMessage和PostMessage的更多相关文章

  1. [转] - SendMessage、PostMessage原理

    SendMessage.PostMessage原理 本文讲解SendMessage.PostMessage两个函数的实现原理,分为三个步骤进行讲解,分别适合初级.中级.高级程序员进行理解,三个步骤分别 ...

  2. SendMessage、PostMessage原理

    SendMessage.PostMessage原理 本文讲解SendMessage.PostMessage两个函数的实现原理,分为三个步骤进行讲解,分别适合初级.中级.高级程序员进行理解,三个步骤分别 ...

  3. SendMessage和PostMessage区别以及WPARAM 和 LPARAM区别

    WPARAM 和 LPARAM wParam和lParam 这两个是Win16系统遗留下来的产物,在Win16API中WndProc有两个参数:一个是WORD类型的16位整型变量:另一个是LONG类型 ...

  4. SENDMESSAGE和POSTMESSAGE

    原文:http://www.cnblogs.com/wind-net/archive/2012/11/01/2750123.html SendMessage 和 PostMessage 的区别 1.首 ...

  5. 关于SendMessage和PostMessage的理解的例子

    对于SendMessage 和 PostMessage 平时口头的解释是 SendMessage 发送消息后等待返回, PostMessage 发送消息后立即返回 . 但是这样解释还是不具体,什么叫等 ...

  6. sendmessage和postmessage的区别

    BOOL   PostMessage(          HWND   hWnd,             //   handle   of   destination   window        ...

  7. (转)深入解析SendMessage、PostMessage

        转自:http://blog.csdn.net/xt_xiaotian/article/details/2778689 本文将使用C++语言,在MFC框架的配合下给出PostMessage.S ...

  8. SendMessage 和 PostMessage

    1.首先是返回值意义的区别,我们先看一下 MSDN 里的声明: LRESULT SendMessage( HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lPar ...

  9. MFC的sendmessage和postmessage 以及sendmessagetimeout

    PostMessage只负责将消息放到消息队列中,不确定何时及是否处理,相当于异步操作,执行后马上返回SendMessage要等到受到消息处理的返回码(DWord类型)后才继续,相当于同步操作,一直在 ...

随机推荐

  1. LoRaWAN调试踩坑心得(一)

    先说两句 在调试和移植的过程中 我们经常想用节点去抓上行包 或者去抓下行包 但在抓取的过程中发现,上行包抓取不到到,或是下行包抓取不到,或者是两个都抓取不到,觉得非常的诡异.明明接收频点.BW和SF都 ...

  2. 第一章 .NET基础-C#基础

    一.1.1. 基础语法 一.1.1.1. 注释符 一.1.1.1.1. 注释符的作用 l 注释 l 解释 一.1.1.1.2. C#中的3中注释符 l 单行注释 // l 多上注释 /* 要注释的内容 ...

  3. Java web部分-面试题

    1.Tomcat的优化经验 答:去掉对web.xml的监视,把jsp提前编辑成Servlet. 有富余物理内存的情况,加大tomcat使用的jvm的内存 2.Servlet的生命周期 答:servle ...

  4. HDU 5057

    题意略. 开始想开一个三维的树状数组,但是一算空间不够,正解是离线操作,按位来计算,一共是10位,所以总共是扫10遍,第i遍只处理第i位的询问, 注意在修改后,要把当前这个位的值存下来(这就是cur数 ...

  5. tomcat8 编写字符编码Filter过滤器无效问题

    做一个解决全站的字符编码过滤器,过滤器类和配置如下: 过滤器类: package com.charles.web.filter; import java.io.IOException; import ...

  6. [Python] socket发送UDP广播实现聊天室功能

    一.说明 本文主要使用socket.socket发送UDP广播来实现聊天室功能. 重点难点:理解UDP通讯流程.多线程.UDP广播收发等. 测试环境:Win10\Python3.5. 程序基本流程:创 ...

  7. Linux基础命令和文件权限

    Linux命令与文件权限 Linux基础命令   reboot  重启   cd         切换目录   cd ..     回到上一级目录   cd ~      回到主目录   cd /  ...

  8. JVM中class文件探索与解析

    一直想成为一名优秀的架构师的我,转眼已经工作快两年了,对于java内核了解甚少,闲来时间,看看JVM,吧自己的一些研究写下来供大家参考,有不对的地方请指正. 废话不多说,一起来看看JVM中类文件是如何 ...

  9. 学习笔记(一)-PyTorch在Windows环境搭建

    一.安装Anaconda 3.5 Anaconda是一个用于科学计算的Python发行版,支持Linux.Mac和Window系统,提供了包管理与环境管理的功能,可以很方便地解决Python并存.切换 ...

  10. xcode删除一个项目

    退出xcode. 在Finder中删除项目文件夹.