C#線程暫停與繼續解決方案
昨天, 老師要我們每個人交一個關於駭客方面的程式,想了半天發現連接埠掃描工具好像好寫點,從昨天寫到今Apsara Infrastructure Management Framework本快完成了,給大家看下效果,不要笑話我哦哦(~~)
圖1 連接埠掃描器
這個程式裡面有一些地方不太滿意,就是掃描的暫停與繼續實現時,使用的是Thread.Suspend和Thread.Resume而這兩個方法,在VS2010裡提示已經過時,不建議使用,在網上查閱了一些資料,發現有個事件通知的方法很好,事件通知的大致原理是,線程在執行過程中暫停,等到其他線程通知時才繼續執行下去,這樣的確是可以起到暫停與繼續的效果。但是,這種暫停是被動的,我需要的是主動暫停,即點下按鈕,線程暫停,再點下按鈕,線程繼續執行。
最終,我想了一種比較另類的方法,大致思路如下:還是採用事件通知的方式,線上程中等待通知,直到來通知了才繼續執行,而主線程(表單線程)中使用一個計時器System.Windows.Forms.Timer 來不停的通知線程,如果計時器間隔時間設定的足夠小,基本上看不出停頓。此時,程式的暫停與繼續實現就很簡單了,相信大家已經想到了,只要在通過控制計時器的Stop()和Start()就可控制線程的暫停與繼續了。
下面是一個下的demo:
運行截圖:
圖2 demo運行效果
C#原始碼:
using System;using System.Windows.Forms;using System.Threading;namespace 線程暫停與繼續實現{ public partial class Form1 : Form { //計時器 private System.Windows.Forms.Timer tm = new System.Windows.Forms.Timer(); //自動重設事件類別 //主要用到其兩個方法 WaitOne() 和 Set() , 前者阻塞當前線程,後者通知阻塞線程繼續往下執行 AutoResetEvent autoEvent = new AutoResetEvent(false); public Form1() { InitializeComponent(); ProgressBar.CheckForIllegalCrossThreadCalls = false; tm.Interval = 1; tm.Tick += new EventHandler(tm_Tick); } //計時器 事件 void tm_Tick(object sender, EventArgs e) { autoEvent.Set(); //通知阻塞的線程繼續執行 } //啟動 private void btnStart_Click(object sender, EventArgs e) { tm.Start(); Thread t = new Thread(DoWork); t.Start(); } //線上程中執行的方法 private void DoWork() { while (progressBar1.Value < progressBar1.Maximum) { progressBar1.PerformStep(); autoEvent.WaitOne(); //阻塞當前線程,等待通知以繼續執行 } } //暫停 private void btnSuspend_Click(object sender, EventArgs e) { tm.Stop(); } //繼續 private void btnResume_Click(object sender, EventArgs e) { tm.Start(); } }}