多线程编程之三——线程间通讯

作者:韩耀旭

原文地址:http://www.vckbase.com/document/viewdoc/?id=1707

七、线程间通讯

  一般而言,应用程序中的一个次要线程总是为主线程执行特定的任务,这样,主线程和次要线程间必定有一个信息传递的渠道,也就是主线程和次要线程间要进行通信。这种线程间的通信不但是难以避免的,而且在多线程编程中也是复杂和频繁的,下面将进行说明。

  1. 使用全局变量进行通信

    由于属于同一个进程的各个线程共享操作系统分配该进程的资源,故解决线程间通信最简单的一种方法是使用全局变量。对于标准类型的全局变量,我们建议使用volatile 修饰符,它告诉编译器无需对该变量作任何的优化,即无需将它放到一个寄存器中,并且该值可被外部改变。如果线程间所需传递的信息较复杂,我们可以定义一个结构,通过传递指向该结构的指针进行传递信息。
     

  2. 使用自定义消息

    我们可以在一个线程的执行函数中向另一个线程发送自定义的消息来达到通信的目的。一个线程向另外一个线程发送消息是通过操作系统实现的。利用Windows操作系统的消息驱动机制,当一个线程发出一条消息时,操作系统首先接收到该消息,然后把该消息转发给目标线程,接收消息的线程必须已经建立了消息循环。

例程7 MultiThread7

  该例程演示了如何使用自定义消息进行线程间通信。首先,主线程向CCalculateThread线程发送消息WM_CALCULATE,CCalculateThread线程收到消息后进行计算,再向主线程发送WM_DISPLAY消息,主线程收到该消息后显示计算结果。

  1. 建立一个基于对话框的工程MultiThread7,在对话框IDD_MULTITHREAD7_DIALOG中加入三个单选按钮IDC_RADIO1,IDC_RADIO2,IDC_RADIO3,标题分别为1+2+3+4+......+10,1+2+3+4+......+50,1+2+3+4+......+100。加入按钮IDC_SUM,标题为“求和”。加入标签框IDC_STATUS,属性选中“边框”;
  2. 在MultiThread7Dlg.h中定义如下变量:
    protected: 	int nAddend; 

    代表加数的大小。

    分别双击三个单选按钮,添加消息响应函数:

    1. void CMultiThread7Dlg::OnRadio1()
    2. {
    3. nAddend=10;
    4. }
    5. void CMultiThread7Dlg::OnRadio2()
    6. {
    7. nAddend=50;
    8. }
    9. void CMultiThread7Dlg::OnRadio3()
    10. {
    11. nAddend=100;
    12. }
    并在OnInitDialog函数中完成相应的初始化工作:
    [c-sharp] view plaincopy
    1. BOOL CMultiThread7Dlg::OnInitDialog()
    2. {
    3. ……
    4. ((CButton*)GetDlgItem(IDC_RADIO1))->SetCheck(TRUE);
    5. nAddend=10;
    6. ……

    在MultiThread7Dlg.h中添加:

    1. #include "CalculateThread.h"
    2. #define WM_DISPLAY WM_USER+2
    3. class CMultiThread7Dlg : public CDialog
    4. {
    5. // Construction
    6. public:
    7. CMultiThread7Dlg(CWnd* pParent = NULL); // standard constructor
    8. CCalculateThread* m_pCalculateThread;
    9. ……
    10. protected:
    11. int nAddend;
    12. LRESULT OnDisplay(WPARAM wParam,LPARAM lParam);
    13. ……
    14. 在MultiThread7Dlg.cpp中添加:
    15. BEGIN_MESSAGE_MAP(CMultiThread7Dlg, CDialog)
    16. ……
    17. ON_MESSAGE(WM_DISPLAY,OnDisplay)
    18. END_MESSAGE_MAP()
    19. LRESULT CMultiThread7Dlg::OnDisplay(WPARAM wParam,LPARAM lParam)
    20. {
    21. int nTemp=(int)wParam;
    22. SetDlgItemInt(IDC_STATUS,nTemp,FALSE);
    23. return 0;
    24. }

    以上代码使得主线程类CMultiThread7Dlg可以处理WM_DISPLAY消息,即在IDC_STATUS标签框中显示计算结果。

  3. 双击按钮IDC_SUM,添加消息响应函数:
    1. void CMultiThread7Dlg::OnSum()
    2. {
    3. m_pCalculateThread=
    4. (CCalculateThread*)AfxBeginThread(RUNTIME_CLASS(CCalculateThread));
    5. Sleep(500);
    6. m_pCalculateThread->PostThreadMessage(WM_CALCULATE,nAddend,NULL);
    7. }
    OnSum()函数的作用是建立CalculateThread线程,延时给该线程发送WM_CALCULATE消息。
  4. 右击工程并选中“New Class…”为工程添加基类为 CWinThread 派生线程类 CCalculateThread。

    在文件CalculateThread.h 中添加

    1. #define WM_CALCULATE WM_USER+1
    2. class CCalculateThread : public CWinThread
    3. {
    4. ……
    5. protected:
    6. afx_msg LONG OnCalculate(UINT wParam,LONG lParam);
    7. ……
    8. 在文件CalculateThread.cpp中添加
    9. LONG CCalculateThread::OnCalculate(UINT wParam,LONG lParam)
    10. {
    11. int nTmpt=0;
    12. for(int i=0;i<=(int)wParam;i++)
    13. {
    14. nTmpt=nTmpt+i;
    15. }
    16. Sleep(500);
    17. ::PostMessage((HWND)(GetMainWnd()->GetSafeHwnd()),WM_DISPLAY,nTmpt,NULL);
    18. return 0;
    19. }
    20. BEGIN_MESSAGE_MAP(CCalculateThread, CWinThread)
    21. //{{AFX_MSG_MAP(CCalculateThread)
    22. // NOTE - the ClassWizard will add and remove mapping macros here.
    23. //}}AFX_MSG_MAP
    24. ON_THREAD_MESSAGE(WM_CALCULATE,OnCalculate)
    25. //和主线程对比,注意它们的区别
    26. END_MESSAGE_MAP()

    在CalculateThread.cpp文件的开头添加一条:

    #include "MultiThread7Dlg.h" 

      以上代码为 CCalculateThread 类添加了 WM_CALCULATE 消息,消息的响应函数是 OnCalculate,其功能是根据参数 wParam 的值,进行累加,累加结果在临时变量nTmpt中,延时0.5秒,向主线程发送WM_DISPLAY消息进行显示,nTmpt作为参数传递。

编译并运行该例程,体会如何在线程间传递消息。

(未完待续)

from:http://blog.csdn.net/yanpingsz/article/details/5891693

C++多线程编程(三)线程间通信的更多相关文章

  1. Java多线程编程(6)--线程间通信(下)

      因为本文的内容大部分是以生产者/消费者模式来进行讲解和举例的,所以在开始学习本文介绍的几种线程间的通信方式之前,我们先来熟悉一下生产者/消费者模式.   在实际的软件开发过程中,经常会碰到如下场景 ...

  2. Java多线程编程核心技术---线程间通信(二)

    通过管道进行线程间通信:字节流 Java提供了各种各样的输入/输出流Stream可以很方便地对数据进行操作,其中管道流(pipeStream)是一种特殊的流,用于在不同线程间直接传送数据,一个线程发送 ...

  3. Java多线程编程核心技术---线程间通信(一)

    线程是操作系统中独立的个体,但这些个体如果不经过特殊处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一.线程间通信可以使系统之间的交互性更强大,在大大提高CPU利用率的同时还会使程序员对各 ...

  4. Java多线程编程(5)--线程间通信

    一.等待与通知   某些情况下,程序要执行的操作需要满足一定的条件(下文统一将其称之为保护条件)才能执行.在单线程编程中,我们可以使用轮询的方式来实现,即频繁地判断是否满足保护条件,若不满足则继续判断 ...

  5. iOS多线程编程之线程间的通信(转载)

    一.简单说明 线程间通信:在1个进程中,线程往往不是孤立存在的,多个线程之间需要经常进行通信 线程间通信的体现 1个线程传递数据给另1个线程 在1个线程中执行完特定任务后,转到另1个线程继续执行任务 ...

  6. java多线程同步以及线程间通信详解&消费者生产者模式&死锁&Thread.join()(多线程编程之二)

    本篇我们将讨论以下知识点: 1.线程同步问题的产生 什么是线程同步问题,我们先来看一段卖票系统的代码,然后再分析这个问题: package com.zejian.test; /** * @author ...

  7. [javaSE] 并发编程(线程间通信)

    新建一个资源类Resource 定义成员变量String name 定义成员变量int age 新建一个输入类Input,实现Runnable接口 定义一个构造方法Input(),传入参数:Resou ...

  8. 《JAVA多线程编程核心技术》 笔记:第三章:线程间通信

    一. 等待/通知机制:wait()和notify()1.1.使用的原因:1.2 具体实现:wait()和notify()1.2.1 方法wait():1.2.2 方法notify():1.2.3 wa ...

  9. Java多线程编程(三)线程间通信

    线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一,可以说,使线程间进行通信后,系统之间的交互性会更强大,在大大提高CPU利用率的同时 ...

随机推荐

  1. title:EL表达式获取Map里面的数值失败的问题

    在控制器中定义了一个Map<Integer,String>集合,看似没有问题,将这个集合的对象map传递到一个JSP页面中,我们都知道,用EL表达式 ${map[key]}就可以取得key ...

  2. 前端开发面试题收集 JS

    前端开发面试题收集-JS篇 收集经典的前端开发面试题 setTimeout的时间定义为0有什么用? javascript引擎是单线程处理任务的,它把任务放在队列中,不会同步执行,必须在完成一个任务后才 ...

  3. python 网络编程第四版

    使用SocketServer 模块来完成服务端编程 1.服务端代码如下: #!/usr/bin/python #!coding:utf-8 import SocketServer as sockets ...

  4. Linux下对字符串进行MD5加密

    Linux下对字符串进行MD5加密 比如要用MD5在linux下加密字符串“test",可以使用命令:$ echo -n test|md5sum098f6bcd4621d373cade4e8 ...

  5. 很详细、很移动的Linux makefile教程:介绍,总述,书写规则,书写命令,使用变量,使用条件推断,使用函数,Make 的运行,隐含规则 使用make更新函数库文件 后序

    很详细.很移动的Linux makefile 教程 内容如下: Makefile 介绍 Makefile 总述 书写规则 书写命令 使用变量 使用条件推断 使用函数 make 的运行 隐含规则 使用m ...

  6. HDU 5700 区间交(线段树)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=5700 [题目大意] 给出一个长度为n的数列和m个区间,现在求k个区间,使得他们的区间交内的数列项和 ...

  7. linux 和unix 的区别

    Linux与Unix的区别  某些PC机的Unix和Linux在实现方面相类似.几乎所有的商业Unix版本都基本支持同样的软件.程序设计环境和网络特性.然而,Linux和Unix的商业版本依然存在许多 ...

  8. Android UI ActionBar功能-ActionBarProvider的使用

    分享功能是很多App都有一个功能,ActionBarProvider可以实现分享功能: 3.0以前的版 本和3.0以后的版 本的区别: public class MainActivity extend ...

  9. 30分钟学会使用grunt打包前端代码【mark】

    grunt 是一套前端自动化工具,一个基于nodeJs的命令行工具,一般用于:① 压缩文件② 合并文件③ 简单语法检查 对于其他用法,我还不太清楚,我们这里简单介绍下grunt的压缩.合并文件,初学, ...

  10. nginx源代码分析--配置文件解析

    ngx-conf-parsing 对 Nginx 配置文件的一些认识: 配置指令具有作用域,分为全局作用域和使用 {} 创建其他作用域. 同一作用域的不同的配置指令没有先后顺序:同一作用域能否使用同样 ...