C++第五十二篇 -- 多线程之消息传递

Posted o云淡风轻o

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++第五十二篇 -- 多线程之消息传递相关的知识,希望对你有一定的参考价值。

主线程向子线程发送消息

参考链接:https://www.cnblogs.com/ranjiewen/p/5729539.html

1. 创建线程语句

    HANDLE hThread;
    DWORD dwThreadId[3];
    for (int i = 0; i < 3; i++) {
        hThread = CreateThread(NULL, 0, FunProc, &i, 0, &dwThreadId[i]);
        CloseHandle(hThread);
    }

 

2. 向子线程发送消息语句。

  a. 在.cpp最上面定义#define MY_MSG WM_USER+100

        for (int j = 0; j < 3; j++) {
            if (!PostThreadMessage(dwThreadId[j], MY_MSG, 0, 0)) {
                printf("post message failed,errno:%d\\n", ::GetLastError());
            }
        }

 

3.1 线程函数接收消息不阻塞

DWORD WINAPI FunProc(LPVOID lpParameter)
{
    MSG msg;
    int i = 0;
    while (TRUE)
    {
        if (PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE))
        {
            if (msg.message == MY_MSG)
                break;
            TranslateMessage(&msg);
            DispatchMessage(&msg);
        }
        i++;
        printf("i = %d\\n", i);
        Sleep(1000);        
    }
    return 0;
}

 

此时接收消息用的是PeekMessage函数,此函数不会阻塞线程。如果此时有消息传来,那么返回的是TRUE,如果没有消息传来,返回的就是FALSE。这个函数的含义是每隔一秒输出一次i,直到有消息传递给线程,那么线程结束。

3.2 线程接收消息阻塞

DWORD WINAPI FunProc(LPVOID lpParameter)
{
    MSG msg;
    int i = 0;
    BOOL stop_thread = FALSE;
    while (!stop_thread)
    {
        if (GetMessage(&msg, 0, 0, 0)) {
            switch (msg.message)
            {
            case MY_MSG:
                stop_thread = TRUE;
                break;
            }
        }
        i++;
        printf("i = %d\\n", i);
        Sleep(1000);        
    }
    return 0;
}

 

此时接收消息用的是GetMessage函数,此函数会阻塞在if语句那儿,直到有消息传来,才会继续下去,所以这个线程只会输出一次i就结束了。

SubWin1.cpp

// SubWin1.cpp : implementation file
//

#include "stdafx.h"
#include "Project2.h"
#include "SubWin1.h"
#include "afxdialogex.h"
#include "resource.h"


// SubWin1 dialog
#define MY_MSG WM_USER+100
//const int MAX_INFO_SIZE = 20;

IMPLEMENT_DYNAMIC(SubWin1, CDialog)

SubWin1::SubWin1(CWnd* pParent /*=nullptr*/)
    : CDialog(IDD_SubWin1, pParent)
{

}

SubWin1::~SubWin1()
{
}

void SubWin1::DoDataExchange(CDataExchange* pDX)
{
    CDialog::DoDataExchange(pDX);
}


BEGIN_MESSAGE_MAP(SubWin1, CDialog)
    ON_WM_TIMER()
    ON_WM_CLOSE()
END_MESSAGE_MAP()


// SubWin1 message handlers

DWORD WINAPI FunProc(LPVOID lpParameter)
{
    MSG msg;
    //DWORD id_thread = GetCurrentThreadId();
    //printf("id_thread = %d\\n", id_thread);
    int i = 0;
    BOOL stop_thread = FALSE;
    while (!stop_thread)
    {
        //if (PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE))
        //{
        //    if (msg.message == MY_MSG)
        //        break;
        //    TranslateMessage(&msg);
        //    DispatchMessage(&msg);
        //}
        if (GetMessage(&msg, 0, 0, 0)) {
            switch (msg.message)
            {
            case MY_MSG:
                stop_thread = TRUE;
                break;
            }
        }
        i++;
        printf("i = %d\\n", i);
        Sleep(1000);        
    }
    return 0;
}


BOOL SubWin1::OnInitDialog()
{
    CDialog::OnInitDialog();

    // TODO:  Add extra initialization here

    HANDLE hThread;
    for (int i = 0; i < 3; i++) {
        hThread = CreateThread(NULL, 0, FunProc, &i, 0, &dwThreadId[i]);
        CloseHandle(hThread);
    }
    

    SetTimer(0, 1000, NULL); //设置一秒刷新一次

    return TRUE;  // return TRUE unless you set the focus to a control
                  // EXCEPTION: OCX Property Pages should return FALSE
}


void SubWin1::OnTimer(UINT_PTR nIDEvent)
{
    // TODO: Add your message handler code here and/or call default
    escape_time++;
    if (escape_time > 10) {
        for (int j = 0; j < 3; j++) {
            if (!PostThreadMessage(dwThreadId[j], MY_MSG, 0, 0)) {
                printf("post message failed,errno:%d\\n", ::GetLastError());
            }
        }
        EndDialog(0x00);
    }

    CDialog::OnTimer(nIDEvent);
}


void SubWin1::OnClose()
{
    // TODO: Add your message handler code here and/or call default
    for (int j = 0; j < 3; j++) {
        if (!PostThreadMessage(dwThreadId[j], MY_MSG, 0, 0)) {
            printf("post message failed,errno:%d\\n", ::GetLastError());
        }
    }

    CDialog::OnClose();
}
View Code

 

这个程序起源于创建一个Dll带MFC库,前面有讲过这一章节内容。

 

以上是关于C++第五十二篇 -- 多线程之消息传递的主要内容,如果未能解决你的问题,请参考以下文章

秒杀多线程第十二篇 多线程同步内功心法——PV操作上

第十二篇:多任务之协程

奋斗吧,程序员——第五十二章 我欲与君相知,长命无绝衰

WPF学习第五十二章 动画性能

第十二篇:线程和进程

第五十二课 linux操作系统原理虚拟机基础原理