一、概述

每个系统中都有线程(至少都有一个主线程),而线程最重要的作用就是并行处理,提高软件的并发率。针对界面来说,还能提高界面的响应能力。一般的,为了应用的稳定性,在数据处理等耗时操作会单独在一个线程中运行,而所有与主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. JVM(五) class类文件的结构

    概述 class类文件的结构可见下面这样图(出处见参考资料),可以参照下面的例子,对应十六进制码,找出找出相应的信息. 其中u2 , u4 表示的意思是占用两个字节和占用四个字节,下面我们将会各项说明 ...

  2. django通用分页封装

    __author__ = 'Administrator'from django.utils.safestring import mark_safe class Page:    def __init_ ...

  3. Hibernate 学习(三)

    一.关系映射 实体类之间的关联映射以及表之间的关系是 ORM 的灵魂之处.对象间的关系的子集可以用下列四种方式解释.关联映射可以是单向的也可以是双向的.  映射类型 描述 Many-to-One 使用 ...

  4. php foreach遍历

    foreach($facility_list['data'] as $facility){ //处理语句} 第一种格式遍历给定的 array_expression_r_r 数组.每次循环中,当前单元的 ...

  5. Java集合 之List(ArrayList、LinkedList、Vector、Stack)理解(new)

    一. ArrayList底层实现原理 对比 和Vector不同,ArrayList中的操作不是线程安全的!所以,建议在单线程中才使用ArrayList,而在多线程中可以选择Vector或者CopyOn ...

  6. pom文件解析

    Maven的依赖是使用Maven坐标来定位的,而Maven坐标主要由GAV(groupId, artifactId, version)构成.因此,使用任何一个依赖之间,你都需要知道它的Maven坐标. ...

  7. thinkphp更新数据库的时候where('')为字符串

    if($user->where('phone='.$phone)->save($dataList)){} if($user->where(array('phone' =>$ph ...

  8. v-model的双向数据绑定(表单)

    可以用 v-model 指令在表单 <input>.<textarea> 及 <select> 元素上创建双向数据绑定.它会根据控件类型自动选取正确的方法来更新元素 ...

  9. Django后台注册

  10. Session、Cookie详解(2)

    session是web开发里一个重要的概念,在大多数web应用里session都是被当做现成的东西,拿来就直接用,但是一些复杂的web应用里能拿来用的session已经满足不了实际的需求,当碰到这样的 ...