Cannot access a disposed object.问题根源所在.....

转自:http://xdeduzb.blog.163.com/blog/static/819936372010417557105/http://

带有UI的C#程序在初始化界面或者由用户触发某一UI更新的时候常常会遇到这样的JIT异常:

System.ObjectDisposedException: Cannot access a disposed object.

Object name: 'XXXX'.

从字面上理解,就是无法得到一个已经被终止的对象。那么在什么情况下这样的事情会发生呢?

看一段代码就能够理解了。

 


Code
           public AuthenticationForm()
            {
                // Required for Windows Form Designer support

                InitializeComponent();

                // TODO: Add any constructor code after InitializeComponent call

                this.action = FormAction.View;

                AppButtons.AppParent = this;

                this.EDITCTLS = new Control[] {RoleDGrid, RegionDGrid, UserNameTBox, CheckAllButton, ClearAllButton, ActiveCkBox, RoleCheckAllButton, RoleClearAllButton};

                this.Initialization();
                BeginAuth("GetUser", new object[]{AddUserButton});

            }


Main方法执行了MainForm的初始化,完成必要组件实例化和数据导入,开始等待用户输入。

Ok,用户这时候点击了一个登录菜单,以上form constructor代码经事件触发,开始执行,啪啪啪完成新form的初始化,最后一步看到BeginAuth方法,它要完成对用户的验证,如果验证通过,登录Form要显示,如果验证失败,主进程要返回错误信息,终止此form的所有资源。

但请注意,这个BeginAuth可不能随便写写。看看这段BeginAuth的实现吧:

     

 

 DataManager.SendAsyncWSRequest(this, this.displayDelegate, spName, DataManager.XMLDOC.InnerXml, hash);
DataManager是数据底层传输处理的接口,它是通用的。DataManager要发出WebService的数据请求,然后获得回答(response),再通过this传入的CallingForm实例调用Form的另一个方法EndAuth。

callingForm.EndAuth(resultXML, spName, hash, new ResponseArgs(false, errorNode.InnerXml, AsyncFailType.WSCaught));
EndAuth要解析WebService的response,判断是否验证通过,和相应Form的资源如何响应。以下是简单的EndAuth实现:

 

Code
         if(resp.Success == false)
          {
                MessageBox.Show(re.Msg, spName + ": WebService call fails");

                base.EndAuth(re.Msg, spName, hash, re);

                this.Close();

                return;

           }

           else

                UpdateDisplay();


Form在失败验证之后会被马上close掉,换句话说,this.Close()会终止form之前初始化的所有组件,GC不知不觉开始回收内存。。。。

本文的主题在这个return之后发生了。return一完成,它退到哪儿了?对,之前form constructor的BeginAuth之后,也就是说,form还没有“出生”就已经被“堕”了。但被“堕”不等于什么也没有(null),毕竟“尸骨”犹在。MainForm在得知sub-form constructor返回以后就会执行类似显示form的方法Show()。要秀就要拿到form句柄,但老子(mainform)拿到的却是一个夭折的孩子。。。当然怎么show也于事无补了。悲惨的Cannot access a disposed object异常就这样发生了。由于是从Mainform触发,它会直接影响主进程,导致程序崩溃。

知道了来龙去脉,那么怎样避免呢?很简单,

1.    不要在constroctor没有做完之前就任意终止资源(原则性)
        拿示例来说的话,就是不要将BeginAuth方法置于构造方法内,置于Form_load()方法中不失为一良策。

2.       在拿来show之前要判断是否为空或已被终止(辅助性)
 

 if (subForm!= null && !subForm.IsDisposed)

    subForm.Show();

 

转者注:我的问题在于开了一个线程在关闭窗口前没有把它关掉,所以我在窗口关闭事件加入线程的Abort(),这个问题就解决了...

你可能感兴趣的:(C#)