昨天,在做一個NPOI讀取的小demo的時候,使用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],還有其他的一些辦法,並沒有解決掉問題。個人的方法或許在大神看來有些麻煩,如果大神有更好的方法,那麼會十分感謝以及歡迎分享在此!!