C#2005後不再支持多線程直接訪問界面的控件(界面創建線程與訪問線程不是同一個線程),不過可以使用delegate來解決:
1. 聲明一個delegate和定義一個delegate的實現函數
view plaincopy to clipboardPRint?
delegate void ShowProgressDelegate(int newPos);
private void ShowProgress(int newPos)
{
// 判斷是否在線程中訪問
if (!_progressBar.InvokeRequired)
{
// 不是的話直接操作控件
_progressBar.Value = newPos;
}
else
{
// 是的話啟用delegate訪問
ShowProgressDelegate showProgress = new ShowProgressDelegate(ShowProgress);
// 如使用Invoke會等到函數調用結束,而BeginInvoke不會等待直接往後走
this.BeginInvoke(showProgress, new object[] { newPos });
}
}
delegate void ShowProgressDelegate(int newPos);
private void ShowProgress(int newPos)
{
// 判斷是否在線程中訪問
if (!_progressBar.InvokeRequired)
{
// 不是的話直接操作控件
_progressBar.Value = newPos;
}
else
{
// 是的話啟用delegate訪問
ShowProgressDelegate showProgress = new ShowProgressDelegate(ShowProgress);
// 如使用Invoke會等到函數調用結束,而BeginInvoke不會等待直接往後走
this.BeginInvoke(showProgress, new object[] { newPos });
}
}
2. 定義線程函數(在另一個線程中可以對界面控件進讀操作)
view plaincopy to clipboardprint?
private void ProgressStart()
{
while (true)
{
int newPos = _progressBar.Value + 10;
if (newPos > _progressBar.Maximum)
{
newPos = _progressBar.Minimum;
}
Trace.WriteLine(string.Format("Pos: {0}", newPos));
// 這裡直接調用方法,由其內部自動判斷是否啟用delegate
ShowProgress(newPos);
Thread.Sleep(100);
}
}
private void ProgressStart()
{
while (true)
{
int newPos = _progressBar.Value + 10;
if (newPos > _progressBar.Maximum)
{
newPos = _progressBar.Minimum;
}
Trace.WriteLine(string.Format("Pos: {0}", newPos));
// 這裡直接調用方法,由其內部自動判斷是否啟用delegate
ShowProgress(newPos);
Thread.Sleep(100);
}
}
3. 線程的啟動和終止
view plaincopy to clipboardprint?
private Thread _progressThread;
_progressThread = new Thread(new ThreadStart(ProgressStart));
// 可選,功用:即使該線程不結束,進程也可以結束
_progressThread.IsBackground = true;
_progressThread.Start();
_progressThread.Abort();
// 可選,功用:等到線程結束才繼續
_progressThread.Join();
_progressThread = null;