c#,c++,qt中多线程访问UI控件线程的问题汇总

2023-05-16

c#和MFC中可以创建UI控件及模态/非模态对话框,Qt中只能在主UI线程中创建访问UI控件。
c#和MFC中可以通过全局变量或指针句柄等方式在工作线程中访问其他UI线程控件,其中c#需要Invoke,mfc中线程不安全可能发生未知异常;Qt中只能在主UI线程中创建访问UI控件,如果通过访问全局UI控件指针方式则线程不安全可能回发生未知异常。间接通过信号槽postEnvet方式。
工作线程创建UI对话框需要考虑工作线程是否有Qt(事件循环)不能创建,MFC(子窗体消息循环)domodal开启消息循环,c#(子窗体是否有消息循环)showdialog开启消息循环,类似消息泵的问题。

C#中子线程访问主UI线程的方法:
1.子线程中调用主UI控件对象然后Invoke,异步委托
2.子线程找到主线程句柄并发送消息,UI线程重写winpro消息循环。
3.设置跨线程安全检查属性为false。
注意:c#中子线程中可以创建UI控件,ui控件时哪个线程创建的则只能在哪个线程上安全访问。
对于windows来说,所有的线程都是一样的,但MFC却把线程区分为两种:用户界面(UI)线程和工作者线程。用户界面线程具有消息循环而工作者线程没有。UI线程可以创建窗口并给这些窗口发送消息,工作者线程执行后台任务,因其不接受用户直接输入蘑菇不需要窗口和消息循环。
MFC中子线程访问主UI线程的方法:(工作线程中可以创建UI控件,主UI线程中控件指针对象理论上可直接被子线程操作,但不是线程安全可能会有未知异常)
1.通过创建子线程的时候向线程创建函数里传送UI句柄作为参数,然后在线程方法中把UI句柄赋值转换为UI控件访问。
2.通过UI线程和主线程两端分别发送消息并重写Winpro消息循环实现线程间交互。
3.将UI控件指针赋值到一个全局变量的UI控件指针变量中,在子线程中访问全局指针(此种方法不是线程安全可能会有未知异常)。
注意:MFC线程内获取主窗口句柄方法:CWnd* h_q = AfxGetApp()->GetMainWnd(); //获取主窗口的句柄
在工作线程中重新获取一遍控件的句柄。
CWnd* edit = AfxGetApp()->GetMainWnd()->GetDlgItem(IDC_EDIT1);//获取主窗口编辑框控件的句柄

工作线程
首先是创建模态的对话框:
要清楚一点:DoModal函数是阻塞的,程序执行到这里就停了,直到把该模态对话框关闭之后,才继续执行。而非模态对话框是不管对话框创建完消没消失,程序都会向下走。

UINT _cdecl  C多线程Dlg::ThreadProc(LPVOID lpParameter) 
{
 C多线程Dlg *pThisDlg = (C多线程Dlg *)lpParameter;//获取当前主对话框的指针,要进行强转
 CTestDlg dlg1;
 dlg1.DoModal();//在线程函数中创建模态的对话框,把dlg对象显示为模态对话框
 return 0;
}
void C多线程Dlg::OnBnClickedBtn()
{
CWinThread *pThread = AfxBeginThread(ThreadProc,this);
}
线程退出了,线程函数就结束了,响应申请的资源就释放了,所以DoModal阻塞在那儿模态对话框不会退出,非模态对话框就自然而然的消失了。

工作线程没有消息循环,对话框界面的刷新工作,包括接受用户的鼠标键盘操作,都不会响应,但我们可以给他一个消息循环,但如果都这样做了,那为什么不去用界面线程呢,界面线程考虑的会更全面。

QT中子线程访问主UI线程的方法:
1.GUI线程和辅助线程
如前所述,每个程序在启动时都有一个线程。这个线程被称为“主线程”(在Qt应用程序中也称为“GUI线程”)。Qt GUI必须在这个线程中运行。所有小部件和几个相关类(例如QPixmap)都不能在辅助线程中工作。辅助线程通常称为“工作线程”,因为它用于从主线程卸载处理工作。
首先,子线程不能创建与UI有关的对象,但是可以这样子做.只能在子线程中发一个信号到主线程中,由主线程创建对话窗口.子线程发完信号后,在子线程中while循环调用事件循环,.对话窗口退出之后,主线程调用接口,结束子线程的while循环!
默认的线程在Qt中称为窗口线程,也叫主线程(UI线程),负责窗口事件处理或者窗口控件数据的更新
子线程负责后台的业务逻辑处理,子线程中不能对窗口对象做任何操作,这些事情都要交给窗口线程处理
主线程和子线程之间如果要进行数据的传递,需要使用Qt中的信号槽机制,或postEnvet
1. 线程与界面组件需要注意的地方
在QThread线程中不能直接创建QWidget之类的界面组件.
因为在QT中,所有界面组件相关的操作都必须在主线程中(也就是GUI thread)
所以, QThread线程不能直接操作界面组件.
2.QThread线程如何操作界面组件-方法1
将多线程类对象封装为GUI界面类的类成员
然后在子线程定义信号函数,通过信号槽机制,向界面组件emit发射信号,从而实现间接操作.
3.QThread线程如何操作界面组件-方法2
使用QApplication::postEvent()实现向界面发送事件,从而能够封装一个自定义类
4.使用Invokes()函数来调用界面组件的信号槽-方法3
一般使用该函数(用来调用对方的私有信号或槽):

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

c#,c++,qt中多线程访问UI控件线程的问题汇总 的相关文章

  • C# 中线程同步使用信号量总结

    所谓线程同步 xff0c 就是多个线程在某个对象上执行等待 xff08 也可理解为锁定该对象 xff09 xff0c 直到该对象被解除锁定 C 中对象的类型分为引用类型和值类型 CLR在这两种类型上的等待是不一样的 我们可以简单地理解为在C
  • 对话框及窗体在多线程中的应用(阻塞和非阻塞)

    方式1 阻塞UI线程及消息循环 AutoResetEvent autoResetEvent 61 new AutoResetEvent false Task Factory StartNew 61 gt Form form 61 new F
  • Windows11装新环境问题

    系统重装 1 下载最新的老毛桃U盘启动盘制作工具 xff08 旧版的会不支持windows11启动项 xff09 xff0c 制作U盘启动盘 2 将电脑的bitLocker硬盘加密取消掉在所有设置安全加密里 xff0c 否则将无法启动提示硬
  • c#引用office组件库迁移源码问题

    1 当程序引用了office组件时 xff0c 需要安装与迁移的源码相同版本的offce Library才行 xff0c 否则会报异常 2 可以下载office三合一版快捷安装 xff0c vs在引用里的com选项卡里会自动识别到相应的of
  • TCP和UDP的发送缓冲区和接收缓冲区内存问题

    TCP协议是作用是用来进行端对端数据传送的 xff0c 那么就会有发送端和接收端 xff0c 在操作系统有两个空间即user space和kernal space 每个Tcp socket连接在内核中都有一个发送缓冲区和接收缓冲区 xff0
  • QtCreator修改项目构建目录

    使用QtCreator编译Qt项目时 xff0c 如有需求修改编译过程文件 xff08 即Makefile o exe等文件 xff09 存放目录 xff0c 简单在工具 gt 选项 gt 构建和运行中修改Default build dir
  • QT中删除信号于槽的连接

    如果是在UI里建立的 xff0c 那就在下面这个函数里删除连接槽函数的对应行 void MainForm qt static metacall QObject o QMetaObject Call c int id void a if c
  • C++位操作中按位置0、置1、取反操作

    一 指定的某一位数置1 宏 define setbit x y x 61 1 lt lt y 二 指定的某一位数置0 宏 define clrbit x y x amp 61 1 lt lt y 三 指定的某一位数取反 宏 define r
  • LED恒流驱动IC汇总

    这几天在找LED恒流驱动芯片 xff0c 无意间在LED网论坛上发现这个帖子 xff0c 分享给大家 xff01 LED恒流IC芯片大盘点 韩国LDT LD1016 16位最大90mA LED屏幕 护栏灯管恒流驱动IC LD1048 48位
  • qt C++中指针自动释放内存及程序中的内存操作、管理

    程序加载到内存后代码存储到代码区 xff0c 并将全局变量 静态变量初始化到全局 静态内存区 xff0c 然后会分配2M左右的栈内存区用于存储局部变量 xff0c 并在运行时根据需要可以在堆内存区 空闲内存区及硬盘的虚拟内存区 申请空间 程
  • sqlite数据库文件提示损坏修复方法

    第1章 说明 1 1 下载SQLite Tools 1 2 运行 2 注意 xff1a 为了方便 xff0c 可把要修复的数据库文件直接放到sqlite3 exe路径下然后运行sqlite3 exe就不用输入具体路径了 sqlite3 ex
  • 常用的dos网络命令总结

    一 ping 主要是测试本机TCP IP协议配置正确性与当前网络现状 ping命令的基本使用格式是 xff1a ping IP地址 主机名 域名 t a n count l size t xff1a 连续对IP地址 主机名 域名执行Ping
  • C#中隐藏窗体并执行窗体逻辑的方法

    c 隐藏窗体方方法 this WindowState 61 FormWindowState Minimized this ShowInTaskbar 61 false base SetVisibleCore true 示例如下 Task F
  • qt中xe运行缺少组件,Qt-c++桌面编程报错:qt.qpa.plugin: Could not find the Qt platform plugin “windows“ in ““,最终解决方案

    报以上错误是因为编译出来的exe程序缺少一些qt的组件 xff0c 需要补全放到exe同级目录下即可 编译库 xff1a Qt GUI xff0c qt5 12 1 软件类型 xff1a Qt application xff0c qt桌面软
  • C#中异步窗体的调用的几种方法

    在子线程里创建的窗体在其他线程里调用的异步调用操作方法 xff1a 1 在另一个线程里找到异步窗体句柄 xff0c 向其传送自定义消息ID 异步窗体的消息循环需要重写加入相应自定义方法 2 找到异步线程里的窗体 xff0c 用该窗体类型直接
  • C++中UDP通讯详解

    C 43 43 Socket编程及TCP UDP通信代码实现 一 简介 Socket编程的目的是使网络的进程进行通信 xff0c 基于TCP IP协议簇 xff0c 通过三元组 xff08 ip地址 协议 端口 xff09 标志进程 xff
  • UDP中connect()和send()函数详解

    UDP也是有connect方法的 xff0c 下面为大家讲解一下这个connect方法 我们知道UDP是无连接的 xff0c 它可以给多个IP发送数据包 xff0c 包括广播地址或者多播通信的实现 xff0c 而这些是TCP IP无法实现的
  • UDP传输报文大小详解

    UDP的传输方式 xff1a 面向报文 面向报文的传输方式决定了UDP的数据发送方式是一份一份的 xff0c 也就是应用层交给UDP多长的报文 xff0c UDP就照样发送 xff0c 即一次发送一个报文 那么UDP的报文大小由哪些影响因素
  • UDP通信概念问题

    1 UDP 多个线程 通过同一个端口接收和发送数据 xff0c 是双向不收影响的还是在时序上错开的 xff1f 如题 xff0c 比如接收到了不同地址的的UDP数据 xff0c 现在通过多个线程从同一个端口回发数据 那么数据的收发都是同一个

随机推荐