[转]MFC子线程中更新控件内容的两种办法
一、概述
每个系统中都有线程(至少都有一个主线程),而线程最重要的作用就是并行处理,提高软件的并发率。针对界面来说,还能提高界面的响应能力。一般的,为了应用的稳定性,在数据处理等耗时操作会单独在一个线程中运行,而所有与主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子线程中更新控件内容的两种办法的更多相关文章
- C#子线程中更新ui
本文实例总结了C#子线程更新UI控件的方法,对于桌面应用程序设计的UI界面控制来说非常有实用价值.分享给大家供大家参考之用.具体分析如下: 一般在winform C/S程序中经常会在子线程中更新控件的 ...
- android 不能在子线程中更新ui的讨论和分析
问题描写叙述 做过android开发基本都遇见过 ViewRootImpl$CalledFromWrongThreadException,上网一查,得到结果基本都是仅仅能在主线程中更改 ui.子线程要 ...
- 使用Handler在子线程中更新UI
Android规定仅仅能在主线程中更新UI.假设在子线程中更新UI 的话会提演示样例如以下错误:Only the original thread that created a view hierach ...
- Android多线程之(一)View.post()源码分析——在子线程中更新UI
提起View.post(),相信不少童鞋一点都不陌生,它用得最多的有两个功能,使用简便而且实用: 1)在子线程中更新UI.从子线程中切换到主线程更新UI,不需要额外new一个Handler实例来实现. ...
- VC线程中操作控件,引起程序卡死的问题。
[问题还原] 线程中操作控件,具体为控制一个按键的使能,使能后结束线程. 主程序中有一个死循环,等待线程结束. 然后,就没有然后了-- [解决方案] 在主程序死循环中,如果检测到界面消息,优先处理掉.
- Android在子线程中更新UI(二)
MainActivity如下: package cc.testui2; import android.os.Bundle; import android.view.View; import andro ...
- Android在子线程中更新UI(一)
MainActivity如下: package cc.testui1; import android.os.Bundle; import android.os.Handler; import andr ...
- Android--Handler的用法:在子线程中更新界面
本文主要介绍Android的Handler的用法.Handler能够发送Messsage和Runnable对象到与其相关联的线程的消息队列. 每一个Handler对象与创建它的线程相关联.而且每一个H ...
- 如何在子线程中更新UI
一:报错情况 android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that creat ...
随机推荐
- 06.密封类 sealed
sealed 修饰 密封类是不能被继承的. 密封类是可以去继承别的类. namespace _07.密封类 { class Program { static void Main(string[] ...
- django常用封装
#encoding:utf-8from django.shortcuts import render_to_responseimport hashlibfrom binascii import b2a ...
- java_对象序列化、反序列化
1.概念 序列化:将对象转化为字节序列的过程 反序列化:将字节序列转化为对象的过程 用途: A:将对象转化为字节序列保存在硬盘上,如文件中,如文本中的例子就是将person对象序列化成字节序列,存在p ...
- 基于easyUI实现系统日志管理
此文章是基于 EasyUI+Knockout实现经典表单的查看.编辑 一. 相关文件介绍 1. log.jsp:系统日志管理界面 <!DOCTYPE html PUBLIC "-//W ...
- php+mysql+jquery日历签到
在网站开发过程中我们会经常用到签到功能来奖励用户积分,或者做一些其他活动.这次项目开发过程中做了日历签到,因为没有经验所有走了很多弯路,再次记录过程和步骤. 1.日历签到样式: 2.本次签到只记录本月 ...
- iis添加共享目录为虚拟目录
注意物理路径处不能直接选择映射成的本地盘符!!!
- 理解JS表达式
表达式:是由运算元和运算符(可选)构成,并产生运算结果的语法结构. 基本表达式 以下在ES5中被称为基本表达式(Primary Expression) this.null.arguments等内置的关 ...
- html中块级元素和行内元素
块级元素和行内元素的三个区别 1.行内元素与块级元素直观上的区别: 行内元素会在一条直线上排列,都是同一行,水平方向排列 块级元素独占一行,垂直方向排列.块级元素从新行开始结束接着一个断行 2.块级元 ...
- Java设计模式—原型模式
原型设计模式是一种比较简单的设计模式,在项目中使用的场景非常多. 个人理解: 原型模式实现了对Java中某个对象的克隆功能,即该对象的类必须implements实现Cloneable接口来标识为可被克 ...
- ArcEngnine中IHookHelper的用法
一.IHookHelper 主要在用在自定义类型于AE带的的ICommand或ITool等 1.实例化IHookHelper 对象:IHookHelper m_hookHelper = new Hoo ...