线程间的同信

本文详细阐述了应用程序中线程间通信的方法,包括使用全局变量、自定义消息进行通信。通过实例展示了如何在多线程环境中实现消息传递,以及如何利用消息驱动机制在不同线程间进行数据交换。
一般而言,应用程序中的一个次要线程总是为主线程执行特定的任务,这样,主线程和次要线程间必定有一个信息传递的渠道,也就是主线程和次要线程间要进行通信。这种线程间的通信不但是难以避免的,而且在多线程编程中也是复杂和频繁的,下面将进行说明。


一、使用全局变量进行通信



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


我们可以在一个线程的执行函数中向另一个线程发送自定义的消息来达到通信的目的。一个线程向另外一个线程发送消息是通过操作系统实现的。利用Windows操作系统的消息驱动机制,当一个线程发出一条消息时,操作系统首先接收到该消息,然后把该消息转发给目标线程,接收消息的线程必须已经建立了消息循环。 
例程7 MultiThread7
  该例程演示了如何使用自定义消息进行线程间通信。首先,主线程向CCalculateThread线程发送消息WM_CALCULATE,CCalculateThread线程收到消息后进行计算,再向主线程发送WM_DISPLAY消息,主线程收到该消息后显示计算结果。
建立一个基于对话框的工程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,属性选中“边框”; 
在MultiThread7Dlg.h中定义如下变量: 
protected:
 int nAddend;
代表加数的大小。


分别双击三个单选按钮,添加消息响应函数: 
void CMultiThread7Dlg::OnRadio1() 
{
 nAddend=10;
}


void CMultiThread7Dlg::OnRadio2() 
{
 nAddend=50;
 
}


void CMultiThread7Dlg::OnRadio3() 
{
 nAddend=100;
 
}
并在OnInitDialog函数中完成相应的初始化工作: 
BOOL CMultiThread7Dlg::OnInitDialog()
{
……
 ((CButton*)GetDlgItem(IDC_RADIO1))->SetCheck(TRUE);
 nAddend=10;
……
在MultiThread7Dlg.h中添加: 
#include "CalculateThread.h"
#define WM_DISPLAY WM_USER+2
class CMultiThread7Dlg : public CDialog
{
// Construction
public:
 CMultiThread7Dlg(CWnd* pParent = NULL); // standard constructor
 CCalculateThread* m_pCalculateThread;
……
protected:
 int nAddend;
 LRESULT OnDisplay(WPARAM wParam,LPARAM lParam);
……
在MultiThread7Dlg.cpp中添加: 
BEGIN_MESSAGE_MAP(CMultiThread7Dlg, CDialog)
……
 ON_MESSAGE(WM_DISPLAY,OnDisplay)
END_MESSAGE_MAP()


LRESULT CMultiThread7Dlg::OnDisplay(WPARAM wParam,LPARAM lParam)
{
 int nTemp=(int)wParam;
 SetDlgItemInt(IDC_STATUS,nTemp,FALSE);


  return 0;


}
以上代码使得主线程类CMultiThread7Dlg可以处理WM_DISPLAY消息,即在IDC_STATUS标签框中显示计算结果。 
双击按钮IDC_SUM,添加消息响应函数: 
void CMultiThread7Dlg::OnSum() 
{
 m_pCalculateThread=
  (CCalculateThread*)AfxBeginThread(RUNTIME_CLASS(CCalculateThread));


 Sleep(500);


 m_pCalculateThread->PostThreadMessage(WM_CALCULATE,nAddend,NULL);
}
OnSum()函数的作用是建立CalculateThread线程,延时给该线程发送WM_CALCULATE消息。 
右击工程并选中“New Class…”为工程添加基类为 CWinThread 派生线程类 CCalculateThread。


在文件CalculateThread.h 中添加 
#define WM_CALCULATE WM_USER+1 
class CCalculateThread : public CWinThread
{
……
protected:
 afx_msg LONG OnCalculate(UINT wParam,LONG lParam);
……
在文件CalculateThread.cpp中添加 
LONG CCalculateThread::OnCalculate(UINT wParam,LONG lParam)
{
 int nTmpt=0;
 for(int i=0;i<=(int)wParam;i++)
 {
  nTmpt=nTmpt+i;
 }


 Sleep(500);
    ::PostMessage((HWND)(GetMainWnd()->GetSafeHwnd()),WM_DISPLAY,nTmpt,NULL);


 return 0;
}
BEGIN_MESSAGE_MAP(CCalculateThread, CWinThread)
 //{{AFX_MSG_MAP(CCalculateThread)
  // NOTE - the ClassWizard will add and remove mapping macros here.
 //}}AFX_MSG_MAP
 ON_THREAD_MESSAGE(WM_CALCULATE,OnCalculate)
//和主线程对比,注意它们的区别
END_MESSAGE_MAP()
在CalculateThread.cpp文件的开头添加一条: 
#include "MultiThread7Dlg.h"
  以上代码为 CCalculateThread 类添加了 WM_CALCULATE 消息,消息的响应函数是 OnCalculate,其功能是根据参数 wParam 的值,进行累加,累加结果在临时变量nTmpt中,延时0.5秒,向主线程发送WM_DISPLAY消息进行显示,nTmpt作为参数传递。 
编译并运行该例程,体会如何在线程间传递消息。
### 关于UDL和TCP通 Universal Device Locator (UDL) 是一种用于定位设备的标准协议,通常被用来简化网络中的设备连接过程。对于基于STM32的嵌入式开发环境而言,通过ST-LINK Utility工具可以实现对目标板的调试、下载以及配置操作[^1]。 #### STM32与ST-LINK Utility STM32微控制器系列支持多种外设接口,其中包括UART/SPI/I2C等串行通方式。当涉及到TCP/IP协议栈时,则需借助第三方库或者官方提供的中间件来完成网络功能的支持。例如,使用LwIP轻量级互联网协议栈能够帮助开发者快速搭建起基本的数据传输框架[^2]。 #### 配置与使用说明 为了使能UDP/TCP服务,在项目初始化阶段应该先定义好相应的宏选项并引入必要的头文件;之后按照特定顺序调用函数以建立服务器端口监听或是客户端请求链接的过程。下面给出一段简单的代码片段展示如何创建一个基础的服务程序: ```c #include "lwip/api.h" #include "lwip/sockets.h" void tcp_server_init(void){ int sock; struct sockaddr_in server_addr; /* 创建套接字 */ if((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0){ printf("Failed to create socket\n"); return ; } memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_port = htons(80); server_addr.sin_addr.s_addr = INADDR_ANY; bind(sock,(struct sockaddr*)&server_addr,sizeof(server_addr)); listen(sock ,5); } ``` 以上实例仅作为入门指导用途,请根据实际应用场景调整参数设置及错误处理逻辑等内容[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值