PostMessage/SendMessage在不同线程的调用探究

PostMessage和SendMessage是我们比较常用的windows API,最近也探究这两个api在调用之后,执行的线程问题,发现如下结论:

  1. 仅仅是RegisterClass注册类之后,调用createwindow第一个参数通过:(const TCHAR*)baseClass.classAtom进行传参创建的窗口,并不执行消息循环,然后往这个窗口句柄进行PostMessage和SendMessage发现:PostMessage无效,Post的Message并不会被WndProc接收到,SendMessage可以正常被WndProc接收并处理,执行线程跟SendMessage当前线程为同一个线程。
  2. 在单独的线程RegisterClass注册类,然后createwindow,最后执行消息循环。然后往这个窗口句柄进行PostMessage和SendMessage发现:PostMessage和SendMessage可以正常被WndProc接收并处理,执行线程跟消息循环线程为同一个线程。
  3. 同时如果上面两个代码都同时执行,发现结果一样,也就是说创建的窗口(CreateWindow)在哪个线程,那么消息执行的线程也在该线程。也就是执行线程跟创建窗口的线程保持了一致,无论是SendMessage还是PostMessage。
  4. WM_COPYDATA消息必须是通过SendMessage,如果是PostMessage,消息队列也是收不到的,而且data里面的buffer必须是开辟的内存空间,连字符串常量也不行,否则接收到的是乱码:
        std::wstring strLog = L"data msg";
        wchar_t* pBuffer = new wchar_t[strLog.length() + 2];
        memcpy(pBuffer, strLog.c_str(), 2 * strLog.length() + 2);
        pBuffer[strLog.length()] = 0;
    
        COPYDATASTRUCT cds;
        cds.dwData = 53; // any data
        cds.cbData = 2 * strLog.length() + 1;
        cds.lpData = pBuffer;
        ::SendMessage(MsgIPC.GetHandle(), WM_COPYDATA, 0, (LPARAM)&cds);
        delete pBuffer;
    
    
    // 下面是错误示范:
        LPCTSTR lpszString = L"Hello";
        COPYDATASTRUCT cds;
        cds.dwData = 1; // can be anything 
        cds.cbData = sizeof(TCHAR) * (wcslen(lpszString) + 1);
        cds.lpData = &lpszString;
    
        ::SendMessage(MsgIPC.GetHandle(), WM_COPYDATA, 0, (LPARAM)&cds); 
  5. d
     

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值