windows窗体线程异常_指针在声明和使用时有何不同

windows窗体线程异常_指针在声明和使用时有何不同在多线程设计中,许多人为了省事,会将对话框类或其它类的指针传给工作线程,而在工作线程中调用该类的成员函数或成员变量等等。但是在Debug版本时,在某些情况下,特别是在工作线程中调用pWnd->UpdateData(FALSE)时,会出现错误。这个错误的原因网上有许多地方讲

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺
 在多线程设计中,许多人为了省事,会将对话框类或其它类的指针传给工作线程,而在工作线程中调用该类的成员函数或成员变量等等。

但是在Debug版本时,在某些情况下,特别是在工作线程中调用pWnd->UpdateData(FALSE)时,会出现错误。

这个错误的原因网上有许多地方讲到了,但是,令人失望的是,讲得好的没几篇,都是非要讲什么线程模块状态什么的,让人看得云里雾里(不过,说实话,也就是从这些文章中才知道是怎么回事的)。

其实本人以为,说穿了,很简单,避免多线程冲突,下面举例说明:

在你的对话框类中有一编辑框和一按钮,编辑框关联了变量为m_strText

现在在你按下按钮时,你有代码如下:

m_strText = “Hello”;

UpdateData(FALSE);

在正常情况下你的编辑框中很显然会显示出”Hello”来。

但是,不怕一万,就怕万一,偏偏在你m_strText=”Hello”这个代码执行之后,你的线程切换了,可是在你的工作线程里,你却将m_strText设置成了”Sorry”,结果当线程切换回来后,UpdateData(FALSE)后,编辑框上就变成”Sorry”而不是”Hello”了。

所以,MFC并不建议这种多线程中传递MFC对象的指针,而且MFC人为的加了一个ASSERT_VALID来表示它们的不建议。

但是,不建议并不表示不能用,如果你能够确认你的线程不会互相冲突,你就大胆的用吧

正因为如此,MFC只是在Debug版本中才有这个ASSERT_VALID的问题存在,在Release版本中却没有,因为它没有理由来阻止我们用。

虽然如此,但是毕竟我们的调试许多时候是要用到Debug版本的,MFC的如此做法还是给我们带来了诸多不变,幸运的是,MFC将它的真正检测线程相关MFC对象的代码做成了虚拟函数,也就是说,我们可以重载它,这样在Debug时,也不会出这问题了。

下面,让我们热烈欢迎我们今天的主角出场--
virtual

void
CObject::
AssertValid(

)

const;


ASSERT_VALID最后会调用MFC类对象的AssertValid函数,因此只要重载AssertValid,令其不检测与线程相关的这些东东,就不会弹出出错框了(其实这些出错框,本来就是人为的弹出来的)。

费话就不说了,假设我们的对话框是CTmthDlg,下面是重载后的代码

void CTmthDlg::AssertValid() const

{

    if (m_hWnd == NULL)

        return;     // null (unattached) windows are valid

    // check for special wnd??? values

    ASSERT(HWND_TOP == NULL);       // same as desktop

    if (m_hWnd == HWND_BOTTOM)

        ASSERT(this == &CWnd::wndBottom);

    else if (m_hWnd == HWND_TOPMOST)

        ASSERT(this == &CWnd::wndTopMost);

    else if (m_hWnd == HWND_NOTOPMOST)

        ASSERT(this == &CWnd::wndNoTopMost);

    else

    {

        // should be a normal window

        ASSERT(::IsWindow(m_hWnd));

        // should also be in the permanent or temporary handle map

/*        CHandleMap* pMap = afxMapHWND();

        ASSERT(pMap != NULL);

        CObject* p;

        ASSERT((p = pMap->LookupPermanent(m_hWnd)) != NULL ||

            (p = pMap->LookupTemporary(m_hWnd)) != NULL);

        ASSERT((CWnd*)p == this);   // must be us

*/

        // Note: if either of the above asserts fire and you are

        // writing a multithreaded application, it is likely that

        // you have passed a C++ object from one thread to another

        // and have used that object in a way that was not intended.

        // (only simple inline wrapper functions should be used)

        //

        // In general, CWnd objects should be passed by HWND from

        // one thread to another.  The receiving thread can wrap

        // the HWND with a CWnd object by using CWnd::FromHandle.

        //

        // It is dangerous to pass C++ objects from one thread to

        // another, unless the objects are designed to be used in

        // such a manner.

    }

}

这里我只是简单的从CWnd::AssertValid中拷贝来,然后注释掉检测线程中MFC对象和Windows对象映射的代码。

另外,请注意一下MFC自己的一些注释。

        // It is dangerous to pass C++ objects from one thread to

        // another, unless the objects are designed to be used in

        // such a manner.

现在,请在你的工作线程中调用

((CTmthDlg*)pParam)->UpdateData(FALSE);

然后调试运行,一切工作正常。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/196839.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • C# SM4加密_sql md5加密

    C# SM4加密_sql md5加密usingSystem;usingSystem.Collections.Generic;usingSystem.IO;usingSystem.Linq;usingSystem.Text;usingSystem.Threading.Tasks;namespaceConsoleApp1{publicclassSM4{publicstaticintSM4_ENCRYPT=1;publicstaticintSM4_DECRYPT=0;p

    2022年10月6日
    5
  • 数论题中(杜教筛)交换求和符号

    数论题中(杜教筛)交换求和符号文章目录方阵下三角约数倍数狄利克雷卷积以及杜教筛学习笔记突然对交换求和符号有了新的理解了,用矩阵转置的思路就很好理解,外层循环相当于枚举行,内层枚举列,交换次序就是先枚举列,再枚举行方阵正常的就是∑i=1n∑j=1nf(i,j)=∑j=1n∑i=1nf(i,j)\sum_{i=1}^n\sum_{j=1}^nf(i,j)=\sum_{j=1}^n\sum_{i=1}^nf(i,j)…

    2022年10月12日
    4
  • 2.1.1 操作系统之进程的定义、特征、组成、组织

    文章目录1.进程的定义(1)程序的概念(2)进程的概念(2)进程的定义2.进程的特征3.进程的组成4.进程的组织(1)链接方式(2)索引方式1.进程的定义(1)程序的概念(2)进程的概念一个进程可以包含多个程序(2)进程的定义2.进程的特征3.进程的组成而其中最重要的就是进程控制块PCB(ProcessControlBlock)PCB简介:&nbsp…

    2022年4月13日
    39
  • navicat premium 15 激活码 mac【在线注册码/序列号/破解码】「建议收藏」

    navicat premium 15 激活码 mac【在线注册码/序列号/破解码】,https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月20日
    77
  • 宝塔面板无法卸载php,宝塔面板如何卸载「建议收藏」

    宝塔面板无法卸载php,宝塔面板如何卸载「建议收藏」windows面板卸载1.打开宝塔面板windows版安装目录,路径为:面板安装数据盘:\BtSoft\ServerAdmin2.运行UnInstall.exe开始面板卸载3.最后使用注册表清理软件或者360清理,清理注册表才可以清除服务文件。在卸载完成后,重启服务器以确保卸载干净。linux面板卸载方法一、脚本卸载1.你需要先在面板中将通过面板安装的所有软件卸载,如nginx、mysql、…

    2025年9月21日
    6
  • vue之解决跨域问题[通俗易懂]

    vue之解决跨域问题[通俗易懂]同源策略:http协议、主机名、端口号都要相同。因为浏览器同源策略的影响,向后端服务器请求数据的时候,不能进行访问。可以采用代理服务器的方式,代理服务器:浏览器向一个相同同源策略的g代理服务器上请求资源,因为服务器之间没有同源策略,代理服务器就去找后端服务器请求资源,在返回给浏览器解决方法一:在根目录下新建vue.config.js文件,这里是js文件哈。module.exports={ lintOnSave:false,//取消格式化 devServer:{ proxy:

    2022年9月17日
    1

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号