昨天,在做一个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],还有其他的一些办法,并没有解决掉问题。个人的方法或许在大神看来有些麻烦,如果大神有更好的方法,那么会十分感谢以及欢迎分享在此!!