zoukankan      html  css  js  c++  java
  • C#中有关资源、BeginInvoke, Invoke和事件的事情

     

    事情是这么来的,我开发的一个程序报了一个错误 “在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke错误”。

    然后我在网上查资料,发现一个有意思的问题,文章出处为“在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke”错误。

    问题

    程序是如下这样的。

    Form1有Button1、Button2和Button3两个按钮,Button2是动态的new Form2窗体,Form2中注册了Form1中的事件.

    Button1的作用是关闭所有new出来的Form2窗体,并且把其资源释放掉(Dispose()).

    Button3是让Form2中注册Form1了的事件都发生。

    操作:

    先点Button2  new出几个Form2的窗体,然后点击Button1释放掉所有的资源,然后再单击Button2 new出几个Form2的窗体,再单击Button3问题就出现了。老是提示"在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke"。

    作者给出的解析

    “在Window窗体程序开发的时候,如果使用多线程编程,在子线程中访问主线程窗体内的控件,就需要使用控件的Control.Invoke方法或者BeginInvoke方法。但是有时候因为Window执行速度太快,尤其是你写代码的时候在InitializeComponent();完成之前起了一个线程去执行某些操作,涉及到窗体控件的,当你在调用Control.Invoke的时候,就可能出现 “在创建窗口句柄之前不能在控件上调用 Invoke 或 BeginInvoke”错误。

    这个解释我感觉十分有道理。

    给出的解决方法

    解决的办法就是让线程等待,直到窗口句柄创建完毕:
    //防止在窗口句柄初始化之前就走到下面的代码
    while (!this.IsHandleCreated)
    {
        ;
    }
    this.BeginInvoke(new ProListIndexChangedDelegate(GetProLyric));

    //根据不同情况也可以:
    if (this.IsHandleCreated)
    BeginInvoke(new ProListIndexChangedDelegate(GetProLyric));

     

    其它人(钻葛格)给的解决方法:

    一个更巧妙的方法,只要在BeginInvoke方法的调用语句前再加一句:IntPtr i = this.Handle;就OK了,这比死循环配合this.IsHandleCreated的判断方法更简洁,因为this.Handle这个属性本身就对应一个方法,取不到句柄,程序就不会向下进行。

    如果以上方式时,IntPtr IsHandleCreated = this.Handle;报错:“从不是创建他的线程访问”。则可把this换成你想提取的句柄所在的线程中的窗体类(或其中任一控件)的实例名。

     本人也就是通过该方法最终解决问题的。也很有可能是作者给出的解释的那种原因。

    结果

    作者的解决的结果

    再次新建窗体的窗口句柄在Show()之后都通过IsHandleCreated属性检测过了,再调用Invoke()之前都创建了,还是会出错。那问题出在哪里呢?一开始还以为单击Button1没有真正的释放资源,然后我就强制GC了,然后检测不到以前的窗体了,但是还是不行,还是有问题。

    然后就想到了“事件”上面来了。是不是事件的原因呀?以前创建的窗体没有注销该事件,把之前所有的窗体都注销该事件,问题就解决了......

     

    有留言认为作者的意思是:根本不是没创建的原因,而是没有清空事件。

    我觉得确实应该是访问了没有创建的UI资源造成的!作者的解决方法是对的,但是说法是错的。
    也就是作者在Button1中释放(Close)了Form2了。而却没有取消Form2中所订阅的Form1的事件。
    此时当点击了button3的话,将激发Form1中事件的委托链,而已经释放掉的Form2对象的事件的响应,由于找到不到该对象资源了,造成了老是提示"在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke"。
    解决办法缺失如作者所说,去除已经被释放对象的订阅的事件即可。

     

  • 相关阅读:
    巨蟒python全栈开发-第11阶段 ansible_project4
    正则面试题
    正确的邮件发送格式?
    巨蟒python全栈开发-第11阶段 ansible_project3
    巨蟒python全栈开发-第11阶段 ansible_project2
    项目资源地址
    网络基础练习题
    巨蟒python全栈开发-第11阶段 ansible_project1
    数据库之单表查询
    数据的增删改
  • 原文地址:https://www.cnblogs.com/arxive/p/9805432.html
Copyright © 2011-2022 走看看