.NET中OpenFileDialog使用报线程错误的解决方法

昨天,在做一个NPOI读取的小demo的时候,使用OpenFileDialog打开文件,最开始的写法,直接在按钮点击事件中写,会报错,代码如下:

 1 OpenFileDialog ofd = new OpenFileDialog();
 2 ofd.Filter = "Microsoft Office Excel(*.xls;*.xlsx)|*.xls;*.xlsx";
 3 ofd.FilterIndex = 1;
 4 ofd.RestoreDirectory = true;
 5 
 6 
 7 if (ofd.ShowDialog() == DialogResult.OK)
 8 {
 9       //检测打开文件路径是否为空地址
10       if (!string.IsNullOrEmpty(ofd.FileName))
11       {
12                     ReadFromExcelFile(ofd.FileName);
13       }
14       else
15       {
16                     this.textBox1.Text = "请打开excel文件";
17        }
18 }

 

或者直接

using(OpenFileDialog ofd = new OpenFileDialog()){ ofd.ShowDialog(); }

  

这两种,无论哪种写法,在代码执行的时候,会报错,具体报错为:

“System.Threading.ThreadStateException”类型的未经处理的异常在 System.Windows.Forms.dll 中发生 

其他信息: 在可以调用 OLE 之前,必须将当前线程设置为单线程单元(STA)模式。请确保您的 Main 函数带有 STAThreadAttribute 标记。 只有将调试器附加到该进程才会引发此异常。

 

这种情况,在网上查询,是说线程问题,就是线程冲突了,不知道该执行哪一个,具体说法如下:

COM提供的线程模型共有三种:Single-Threaded Apartment(STA 单线程套间)、Multithreaded Apartment(MTA 多线程套间)和Neutral Apartment/Thread Neutral Apartment/Neutral Threaded Apartment(NA/TNA/NTA 中立线程套间,由COM+提供)。

STA 一个对象只能由一个线程访问,相当于windows的消息循环,实现方式也是通过消息循环的,ActiveX控件、OLE文档服务器等有界面的,都使用STA的套间。 MTA 一个对象可以被多个线程访问,即这个对象的代码在自己的方法中实现了线程保护,保证可以正确改变自己的状态。 

所以创建和访问一个activex或者ole对象时,必须设置线程模式为sta。

 

那么,在子线程中应该如何使用OpenFileDialog才不会继续报这种错误呢,下面就是更改后的代码:

 1         /// <summary>
 2         /// 单线程打开excel文档
 3         /// </summary>
 4         /// <param name="sender"></param>
 5         /// <param name="e"></param>
 6         private void btnXlx_Click(object sender, EventArgs e)
 7         {
 8             this.textBox1.Text = string.Empty;
 9 
10             System.Threading.Thread s = new System.Threading.Thread(new System.Threading.ThreadStart(getExcel));
11             s.ApartmentState = System.Threading.ApartmentState.STA;
12             s.Start();
13 
14         }
15 
16         /// <summary>
17         /// 读取excel文档地址
18         /// </summary>
19         private void getExcel()
20         {
21             OpenFileDialog ofd = new OpenFileDialog();
22             ofd.Filter = "Microsoft Office Excel(*.xls;*.xlsx)|*.xls;*.xlsx";
23             ofd.FilterIndex = 1;
24             ofd.RestoreDirectory = true;
25 
26 
27             if (ofd.ShowDialog() == DialogResult.OK)
28             {
29                 //检测打开文件路径是否为空地址
30                 if (!string.IsNullOrEmpty(ofd.FileName))
31                 {
32                     ReadFromExcelFile(ofd.FileName);
33                 }
34                 else
35                 {
36                     this.textBox1.Text = "请打开excel文件";
37                 }
38             }
39         }

 

就是把线程执行的内容,单独分离出来形成一个方法,然后在事件中编写执行子线程单线程执行语句,这种情况下,就不会在报那种线程异常的错误了。

PS:个人通过搜索网上内容,总结出来的,感觉的可以成解决的一个方法,向其他诸如Main函数前门加[STAThread],还有其他的一些办法,并没有解决掉问题。个人的方法或许在大神看来有些麻烦,如果大神有更好的方法,那么会十分感谢以及欢迎分享在此!!

 

 

posted @ 2018-01-19 09:38  叁水木  阅读(1008)  评论(1编辑  收藏  举报