默认情况下.NET Framework不允许在一个线程中直接访问另一个线程中的控件,这是因为如果又两个或多个线程同时访问某一个控件,则可能会迫使该控件进入一种不确定的状态。甚至可能出现不同线程争用控件引起死锁问题。
但是,为了在窗体上显示线程中处理的信息,我们可能经常需要在一个线程中访问另一个线程中的控件。有两种办法可以实现这个功能,一种是使用委托(delegate)和事件(event)来完成,另一种是利用BackgroundWorker组件来实现这个功能。
为了让不是创建控件的线程共享该控件对象,Windows应用程序的每一个控件都提供一个Invoke方法,该方法利用委托实现其他线程对该控件的操作。例如:
private delegate void AddMessageDelegate(string message);
public void AddMessage(string message)
{
if(richTextBox1.InvokeRequired)
{
AddMessageDelegate d = AddMessage;
richTextBox1.Invoke(d,message);
}
else
{
richTextBox1.AppendText(message);
}
}
class Test
{
public delegate void ThreadStartDelegate(System.Windows.Forms.RichTextBox ric, string msg);
public static void ShowMsg(System.Windows.Forms.RichTextBox ric, string msg)
{
if (!string.IsNullOrEmpty(msg))
{
if (ric.InvokeRequired)//获取一个值,该值指示调用方在对控件进行方法调用时是否必须调用 Invoke 方法,因为调用方位于创建控件所在的线程以外的线程中。
{
ThreadStartDelegate d = new ThreadStartDelegate(ShowMsg);
ric.BeginInvoke(d, ric, msg);//在创建控件的基础句柄所在线程上,用指定的参数异步执行指定委托。
}
else
{
ric.AppendText(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + " " + msg+Environment.NewLine);
ric.ScrollToCaret();
}
}
}
public static void ShowInfo(System.Windows.Forms.RichTextBox ric, string msg)
{
if (!string.IsNullOrEmpty(msg))
{
if (ric.InvokeRequired)
{
ThreadStartDelegate d = new ThreadStartDelegate(ShowInfo);
ric.BeginInvoke(d, ric, msg);
}
else
{
ric.AppendText(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + " 报告单编号为【" + msg +"】的记录上传成功!"+ Environment.NewLine);
ric.ScrollToCaret();
}
}
}
public static void ShowInfo(System.Windows.Forms.RichTextBox ric, string msg,string ex)
{
if (!string.IsNullOrEmpty(msg))
{
if (ric.InvokeRequired)
{
ThreadStartDelegate d = new ThreadStartDelegate(ShowInfo);
ric.BeginInvoke(d, ric, msg);
}
else
{
ric.AppendText(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + " 报告单编号为【" + msg + "】的记录上传失败!"+ex + Environment.NewLine);
ric.ScrollToCaret();
}
}
}
public static bool SysTest(System.Windows.Forms.RichTextBox ric)
{
SqlConnection localconn = ConnManager.GetLoaclConn();
OleDbConnection remoteconn = ConnManager.GetRemoteConn();
if (ConnManager.OpenSqlConn(localconn))
{
ConnManager.CloseSqlConn(localconn);
ShowMsg(ric, "本地数据库连接成功!");
if (ConnManager.OpenOleDbConn(remoteconn))
{
ConnManager.CloseOleDbConn(remoteconn);
ShowMsg(ric, "远程数据库连接成功!");
return true;
}
else
{
ConnManager.CloseOleDbConn(remoteconn);
ShowMsg(ric, DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + " 远程数据库连接失败!");
return false;
}
}
else
{
ShowMsg(ric, "本地数据库连接成功!");
return false;
}
}
}