欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

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

程序员文章站 2022-07-04 21:35:21
昨天,在做一个npoi读取的小demo的时候,使用openfiledialog打开文件,最开始的写法,直接在按钮点击事件中写,会报错,代码如下: openfile...

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

openfiledialog ofd = new openfiledialog();
ofd.filter = "microsoft office excel(*.xls;*.xlsx)|*.xls;*.xlsx";
ofd.filterindex = 1;
ofd.restoredirectory = true;


if (ofd.showdialog() == dialogresult.ok)
{
   //检测打开文件路径是否为空地址
   if (!string.isnullorempty(ofd.filename))
   {
          readfromexcelfile(ofd.filename);
   }
   else
   {
          this.textbox1.text = "请打开excel文件";
    }
} 

或者直接

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才不会继续报这种错误呢,下面就是更改后的代码:

/// <summary>
    /// 单线程打开excel文档
    /// </summary>
    /// <param name="sender"></param>
    /// <param name="e"></param>
    private void btnxlx_click(object sender, eventargs e)
    {
      this.textbox1.text = string.empty;

      system.threading.thread s = new system.threading.thread(new system.threading.threadstart(getexcel));
      s.apartmentstate = system.threading.apartmentstate.sta;
      s.start();

    }

    /// <summary>
    /// 读取excel文档地址
    /// </summary>
    private void getexcel()
    {
      openfiledialog ofd = new openfiledialog();
      ofd.filter = "microsoft office excel(*.xls;*.xlsx)|*.xls;*.xlsx";
      ofd.filterindex = 1;
      ofd.restoredirectory = true;


      if (ofd.showdialog() == dialogresult.ok)
      {
        //检测打开文件路径是否为空地址
        if (!string.isnullorempty(ofd.filename))
        {
          readfromexcelfile(ofd.filename);
        }
        else
        {
          this.textbox1.text = "请打开excel文件";
        }
      }
    }

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

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。