一、概述

每个系统中都有线程(至少都有一个主线程),而线程最重要的作用就是并行处理,提高软件的并发率。针对界面来说,还能提高界面的响应能力。一般的,为了应用的稳定性,在数据处理等耗时操作会单独在一个线程中运行,而所有与主UI线程有关的控件数据刷新应该到主UI线程中处理。也就是数据处理线程发消息,让界面UI去更新控件。在MFC中线程分为界面线程和工作者线程,界面实际就是一个线程画出来的东西,这个线程维护一个“消息队列”,“消息队列”也是界面线程和工作者线程的最大区别,这个词应该进到你的脑子里,根深蒂固的!MFC中有两类线程,分别称之为工作者线程和用户界面线程。二者的主要区别在于工作者线程没有消息循环,而用户界面线程有自己的消息队列和消息循环。

在MFC中,一般用全局函数AfxBeginThread()来创建并初始化一个线程(工作者线程,还有一个重载形式是用于创建用户界面线程)的运行。函数原型:

CWinThread* AfxBeginThread(
AFX_THREADPROC pfnThreadProc,
LPVOID pParam,
int nPriority = THREAD_PRIORITY_NORMAL,
UINT nStackSize = ,
DWORD dwCreateFlags = ,
LPSECURITY_ATTRIBUTES lpSecurityAttrs = NULL
);

返回值: 成功时返回一个指向新线程的线程对象的指针,否则NULL。

pfnThreadProc:线程的入口函数,声明一定要如下: UINT MyThreadFunction(LPVOID pParam),不能设置为NULL。如果是类成员函数,一定要是静态成员函数。

pParam:传递入线程的参数,注意它的类型为:LPVOID,所以我们可以传递一个结构体或者类对象到线程。一般传递this指针,以方便调用类的非静态成员,因为线程函数是静态函数。

nPriority:线程的优先级,一般设置为0,让它和主线程具有共同的优先级。

nStackSize:指定新创建的线程的栈的大小。如果为 0,新创建的线程具有和主线程一样的大小的栈。

dwCreateFlags:指定创建线程以后,线程有怎么样的标志。可以指定两个值:CREATE_SUSPENDED:线程创建以后,会处于挂起状态,直到调用:ResumeThread。0 : 创建线程后就开始运行。

lpSecurityAttrs:指向一个 SECURITY_ATTRIBUTES 的结构体,用它来标志新创建线程的安全性。如果为 NULL,那么新创建的线程就具有和主线程一样的安全性。

常见用法:

AfxBeginThread(MyThreadFunction, this); 

传递线程参数为this,即类本身,是为了能在线程函数中获得类中非静态成员变量,因为线程函数是静态函数。

MFC子线程中更新控件内容有两种方法,一种是在子线程中通过全局函数更新控件内容,一种是在子线程中通过发送自定义消息来更新控件内容。

二、通过全局函数更新控件内容

1.在对话框类CThreadDemoDlg中添加成员变量——线程对象的指针和线程函数

CWinThread *m_pThread;
static UINT ThreadFunction(LPVOID pParam);

2.实现线程函数,使用全局函数::SetWindowText、::GetDlgItem更新控件内容

UINT CThreadDemoDlg::ThreadFunction(LPVOID pParam)
{
CThreadDemoDlg *pDlg = (CThreadDemoDlg *)pParam; while (TRUE) {
::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_STATIC), L"Hello World");
Sleep();
::SetWindowText(::GetDlgItem(pDlg->m_hWnd, IDC_STATIC), L"Hello Android");
Sleep();
}
return ;
}

3.在成员函数OnInitDialog创建线程并启动

m_pThread = AfxBeginThread((AFX_THREADPROC)ThreadFunction, this);   

三、通过发送自定义消息更新控件内容

1.在头文件中定义消息ID

#define WM_UPDATE_STATIC (WM_USER + 100)  

2.在对话框类CThreadDemoDlg中添加成员——线程对象的指针和线程函数

CWinThread *m_pThread;
static UINT ThreadFunction(LPVOID pParam);

3.声明自定义的消息函数

afx_msg LRESULT OnUpdateStatic(WPARAM wParam, LPARAM lParam);  

4.在CPP文件中添加消息映射

BEGIN_MESSAGE_MAP(CThreadDemoDlg, CDialog)
//......
ON_MESSAGE(WM_UPDATE_STATIC, &CThreadDemoDlg::OnUpdateStatic)
//......
END_MESSAGE_MAP()

5.实现自定义消息响应函数

LRESULT CThreadDemoDlg::OnUpdateStatic(WPARAM wParam, LPARAM lParam)
{
if (wParam == ) {
GetDlgItem(IDC_STATIC)->SetWindowText(L"Hello Linux");
} else {
GetDlgItem(IDC_STATIC)->SetWindowText(L"Hello Windows");
} return ;
}

6.实现线程函数,并通过PostMessage发送自定义消息

UINT CThreadDemoDlg::ThreadFunction(LPVOID pParam)
{
CThreadDemoDlg *pDlg = (CThreadDemoDlg *)pParam; while (TRUE) {
::PostMessage(pDlg->m_hWnd, WM_UPDATE_STATIC, , );
Sleep();
::PostMessage(pDlg->m_hWnd, WM_UPDATE_STATIC, , );
Sleep();
} return ;
}

7.在成员函数OnInitDialog创建线程并启动

m_pThread = AfxBeginThread((AFX_THREADPROC)ThreadFunction, this);   

  通过发送自定义消息更新控件内容总体思路:在主界面所在的主线程中创建线程--->将线程的起始函数绑定到类的静态成员方法--->在静态成员方法中调用PostMessage()或SendMessage()发送自定义消息--->在自定义消息的消息响应函数中更新主界面上控件的状态显示。

原文链接:MFC子线程中更新控件内容的两种办法

[转]MFC子线程中更新控件内容的两种办法的更多相关文章

  1. C#子线程中更新ui

    本文实例总结了C#子线程更新UI控件的方法,对于桌面应用程序设计的UI界面控制来说非常有实用价值.分享给大家供大家参考之用.具体分析如下: 一般在winform C/S程序中经常会在子线程中更新控件的 ...

  2. android 不能在子线程中更新ui的讨论和分析

    问题描写叙述 做过android开发基本都遇见过 ViewRootImpl$CalledFromWrongThreadException,上网一查,得到结果基本都是仅仅能在主线程中更改 ui.子线程要 ...

  3. 使用Handler在子线程中更新UI

    Android规定仅仅能在主线程中更新UI.假设在子线程中更新UI 的话会提演示样例如以下错误:Only the original thread that created a view hierach ...

  4. Android多线程之(一)View.post()源码分析——在子线程中更新UI

    提起View.post(),相信不少童鞋一点都不陌生,它用得最多的有两个功能,使用简便而且实用: 1)在子线程中更新UI.从子线程中切换到主线程更新UI,不需要额外new一个Handler实例来实现. ...

  5. VC线程中操作控件,引起程序卡死的问题。

    [问题还原] 线程中操作控件,具体为控制一个按键的使能,使能后结束线程. 主程序中有一个死循环,等待线程结束. 然后,就没有然后了-- [解决方案] 在主程序死循环中,如果检测到界面消息,优先处理掉.

  6. Android在子线程中更新UI(二)

    MainActivity如下: package cc.testui2; import android.os.Bundle; import android.view.View; import andro ...

  7. Android在子线程中更新UI(一)

    MainActivity如下: package cc.testui1; import android.os.Bundle; import android.os.Handler; import andr ...

  8. Android--Handler的用法:在子线程中更新界面

    本文主要介绍Android的Handler的用法.Handler能够发送Messsage和Runnable对象到与其相关联的线程的消息队列. 每一个Handler对象与创建它的线程相关联.而且每一个H ...

  9. 如何在子线程中更新UI

    一:报错情况 android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that creat ...

随机推荐

  1. 06.密封类 sealed

    sealed   修饰 密封类是不能被继承的. 密封类是可以去继承别的类. namespace _07.密封类 { class Program { static void Main(string[] ...

  2. django常用封装

    #encoding:utf-8from django.shortcuts import render_to_responseimport hashlibfrom binascii import b2a ...

  3. java_对象序列化、反序列化

    1.概念 序列化:将对象转化为字节序列的过程 反序列化:将字节序列转化为对象的过程 用途: A:将对象转化为字节序列保存在硬盘上,如文件中,如文本中的例子就是将person对象序列化成字节序列,存在p ...

  4. 基于easyUI实现系统日志管理

    此文章是基于 EasyUI+Knockout实现经典表单的查看.编辑 一. 相关文件介绍 1. log.jsp:系统日志管理界面 <!DOCTYPE html PUBLIC "-//W ...

  5. php+mysql+jquery日历签到

    在网站开发过程中我们会经常用到签到功能来奖励用户积分,或者做一些其他活动.这次项目开发过程中做了日历签到,因为没有经验所有走了很多弯路,再次记录过程和步骤. 1.日历签到样式: 2.本次签到只记录本月 ...

  6. iis添加共享目录为虚拟目录

    注意物理路径处不能直接选择映射成的本地盘符!!!

  7. 理解JS表达式

    表达式:是由运算元和运算符(可选)构成,并产生运算结果的语法结构. 基本表达式 以下在ES5中被称为基本表达式(Primary Expression) this.null.arguments等内置的关 ...

  8. html中块级元素和行内元素

    块级元素和行内元素的三个区别 1.行内元素与块级元素直观上的区别: 行内元素会在一条直线上排列,都是同一行,水平方向排列 块级元素独占一行,垂直方向排列.块级元素从新行开始结束接着一个断行 2.块级元 ...

  9. Java设计模式—原型模式

    原型设计模式是一种比较简单的设计模式,在项目中使用的场景非常多. 个人理解: 原型模式实现了对Java中某个对象的克隆功能,即该对象的类必须implements实现Cloneable接口来标识为可被克 ...

  10. ArcEngnine中IHookHelper的用法

    一.IHookHelper 主要在用在自定义类型于AE带的的ICommand或ITool等 1.实例化IHookHelper 对象:IHookHelper m_hookHelper = new Hoo ...