1
我已经使用C#创建了COM服务器,我的客户端可以接收实时更新。 通常从不同的线程发起更新。 但我注意到,当回调方法更新电子表格时,Excel崩溃。 有什么方法可以在UI线程中调用更新吗?从另一个线程更新Excel电子表格
P.S.我知道Excel的RTD功能。但它不适合我的需要,因为我需要一个更新中的几个参数。
我已经使用C#创建了COM服务器,我的客户端可以接收实时更新。 通常从不同的线程发起更新。 但我注意到,当回调方法更新电子表格时,Excel崩溃。 有什么方法可以在UI线程中调用更新吗?从另一个线程更新Excel电子表格
P.S.我知道Excel的RTD功能。但它不适合我的需要,因为我需要一个更新中的几个参数。
你正在寻找的是对你的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参考......你有退出应用程序之前将他们释放
遗憾没有得到我怎么能达到ISyncronizedInvoke实例? – Vitaliy
只需要一个任意的控制...就像你的主窗体... – DarkSquirrel42
你的意思是创建隐藏窗体吗? – Vitaliy