2013-11-04 45 views
1

我已经使用C#创建了COM服务器,我的客户端可以接收实时更新。 通常从不同的线程发起更新。 但我注意到,当回调方法更新电子表格时,Excel崩溃。 有什么方法可以在UI线程中调用更新吗?从另一个线程更新Excel电子表格

P.S.我知道Excel的RTD功能。但它不适合我的需要,因为我需要一个更新中的几个参数。

回答

1

你正在寻找的是对你的UI线程ISyncronizedInvoke

的调用/ BeginInvoke方法,采取随心所欲的控制,并保持该基准...

从想要触发一个更新的线程,调用调用或BeginInvoke上控制(控制实现ISyncronizedInvoke)与要在UI线程上执行委托......从代表你可以叫你的COM服务器

//编辑:示例代码

using System; 
using System.Windows.Forms; 
using System.Threading; 
using Microsoft.Office.Interop.Excel; 
using System.Runtime.InteropServices; 

namespace WindowsFormsApplication1 
{ 
    public partial class Form1 : Form 
    { 
     Thread someWorkerThread; 

     Microsoft.Office.Interop.Excel.Application ExApp; 
     Worksheet wrkSheet; 

     public Form1() 
     { 
      InitializeComponent(); 

      ExApp = new Microsoft.Office.Interop.Excel.Application(); 

      ExApp.Visible = true; // or else we won't see the window 

      var books = ExApp.Workbooks; 
      var wrkBook = books.Add(); 
      var sheets = wrkBook.Worksheets; 
      wrkSheet = sheets.get_Item(1); 

      Marshal.ReleaseComObject(sheets); 
      Marshal.ReleaseComObject(wrkBook); 
      Marshal.ReleaseComObject(books); 

      someWorkerThread = new Thread(new ParameterizedThreadStart(threadHandler)); 
      someWorkerThread.Start(this); 
     } 

     private void threadHandler(object obj) 
     {// this will be executed on a seperate worker thread 
      Control mainFrm = obj as Control; 
      if (mainFrm == null) 
       throw new ArgumentException("Need to have a Control as parameter"); 
      for (int i = 1; i < 50;i++) 
      { 
       Thread.Sleep(2500); 
       mainFrm.Invoke(new Action<int>(doStuff), i); // this will invoke the main UI thread 
      } 
     } 

     private void doStuff(int i) 
     {// this will be executed on the main UI thread 
      var range = wrkSheet.Range[string.Format("A{0}", i)]; 
      range.Value = "Hello World!"; 
      Marshal.ReleaseComObject(range); 
     } 


     #region designer stuff 
     /// <summary> 
     /// Required designer variable. 
     /// </summary> 
     private System.ComponentModel.IContainer components = null; 

     /// <summary> 
     /// Clean up any resources being used. 
     /// </summary> 
     /// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param> 
     protected override void Dispose(bool disposing) 
     { 
      if (disposing && (components != null)) 
      { 
       components.Dispose(); 
      } 
      base.Dispose(disposing); 
     } 

     #region Windows Form Designer generated code 

     /// <summary> 
     /// Required method for Designer support - do not modify 
     /// the contents of this method with the code editor. 
     /// </summary> 
     private void InitializeComponent() 
     { 
      this.label1 = new System.Windows.Forms.Label(); 
      this.SuspendLayout(); 
      // 
      // label1 
      // 
      this.label1.AutoSize = true; 
      this.label1.Location = new System.Drawing.Point(76, 84); 
      this.label1.Name = "label1"; 
      this.label1.Size = new System.Drawing.Size(149, 13); 
      this.label1.TabIndex = 0; 
      this.label1.Text = "I am an ordinary windows form"; 
      // 
      // Form1 
      // 
      this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); 
      this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font; 
      this.ClientSize = new System.Drawing.Size(284, 262); 
      this.Controls.Add(this.label1); 
      this.Name = "Form1"; 
      this.Text = "Form1"; 
      this.ResumeLayout(false); 
      this.PerformLayout(); 

     } 

     #endregion 

     private System.Windows.Forms.Label label1; 

     #endregion 
    } 
} 

被告知:这个例子不清理wrkSheet参考和ExApp参考......你有退出应用程序之前将他们释放

+0

遗憾没有得到我怎么能达到ISyncronizedInvoke实例? – Vitaliy

+0

只需要一个任意的控制...就像你的主窗体... – DarkSquirrel42

+0

你的意思是创建隐藏窗体吗? – Vitaliy

相关问题