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

原文地址:https://www.cnblogs.com/yzjey/p/8315403.html

时间: 2024-08-24 10:42:53

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

在Idea中hibernateTemplate.find(&quot;from User &quot;);错误提示解决方法

在使用idea进行Java开发时(Springmvc框架下的项目),当使用到hibernateTemplate.find("from User ")时,总是在User下面提示错误,并有红色波浪线,但是并不影响项目正常运行,这里总结两个可能出现这种情况原因: 解决方案一: hibernateTemplate.find("from User ")中的User必须是POJO类,注意首字母大写问题. 解决方案二: 右键项目,选择添加框架,选择Persistence 添加完后

Windows7 64位中出现的KERNELBASE.dll错误的解决方法

最近在服程序时遇到个问题,电脑是win764位,编译完的exe测试,偶尔总报错,报错是偶尔的,有时候报错很频繁,但是有一次测试,测试了半天都没有报错,我以为好,发布输出没一会儿又报错了,真是崩溃了,所有方法我都trycatch了,日志也记录了,也没找定位到哪里报错,这肯定是系统报错的,应该是是我64位编译问题.在网上查找好多资料也,其实这并不是KERNELBASE.dll的问题,也并不需要复制一个KERNELBASE.dll文件替换掉它. 1.0x7513C54F (KernelBase.dll

Java文件中出现这样的提示错误与解决方法:Cannot return from outside a function or method?

1.打开  >>  Myeclipse或Eclipse.(我使用的是Myeclipse) 2.打开  >>  Window  >>  Preferences  如图1: 图1 3.打开  >>  Preferences  >>  MyEclipse  >>  Validation  >>  JavaScript validator for JS files(Bulid中的"√"去掉)  如图2: 图2

PHPExcel在高版本PHP7中,Writer-&gt;save出现ERR_INVALID_RESPONSE错误的解决方法

这个Writer->save错误可能由于很多原因导致,其中有一部分是因为header和缓冲区的错误导致的 这部分具体讨论可以看这里http://stackoverflow.com/questions/8566196/phpexcel-to-download 然而还有一个不容易发现的问题,在高版本PHP7下,出现ERR_INVALID_RESPONSE的错误还可能由于下面的原因导致 1 Fatal error: 'break' not in the 'loop' or 'switch' conte

ORACLE安装过程中检查步骤出现的错误和解决方法【转】

Checking operating system requirements ...Expected result: One of redhat-3,redhat-4,SuSE-9,asianux-1,asianux-2Actual Result: redhat-4Check complete. The overall result of this check is: Passed==========================================================

在mysql中创建存储过程出现1307错误,解决方法

需要删除mysql数据库下proc表 在重新创建 CREATE TABLE `proc` ( `db` char(64) character set utf8 collate utf8_bin NOT NULL default '', `name` char(64) NOT NULL default '', `type` enum('FUNCTION','PROCEDURE') NOT NULL, `specific_name` char(64) NOT NULL default '', `la

VS2010中,出现ERROR MSB8011错误的解决方法

我遇到了这种情况: 昨天写的程序编译连接还好好的,只是简单的改了几句代码,结果今天就显示 C:\Program Files\MSBuild\Microsoft.Cpp\v4.0\Microsoft.CppCommon.targets(738,5):warning MSB3073:命令“"F:\...\Debug\XXX.exe"/RegServer"已退出,... C:\Program Files\MSBuild\Microsoft.Cpp\v4.0\Microsoft.Cp

jQuery中live()使用报错,TypeError: $(...).live is not a function

原地址: https://blog.csdn.net/sdfdyubo/article/details/59536781 1.使用 (1)原写法 /*为选项卡绑定右键*/    $(".tabs li").live('contextmenu', function (e) {        /*选中当前触发事件的选项卡 */        var subtitle = $(this).text();        $('#mainTab').tabs('select', subtitle

bash脚本中出现[[:not found错误的解决方法

bash脚本中出现[[:not found错误的解决方法--bash脚本总结1 今天在写脚本的时候,发生了一个奇怪的问题:在脚本中使用[[的时候报错“[[: not found”.遇到问题自然是解决问题. 1. 使用的bash版本太低? bash --version查看bash版本信息如下 [email protected]:~$bash --version GNU bash, version 3.2.39(1)-release (i486-pc-linux-gnu) Copyright (C)