淺談利用ASP.NET多線程執(zhí)行長(zhǎng)時(shí)間的任務(wù)
在ASP.NET中執(zhí)行一個(gè)長(zhǎng)時(shí)間的操作,有的時(shí)候需要在在客戶端有一個(gè)反饋能了解到任務(wù)的執(zhí)行進(jìn)度,大致看了一下有這么幾種做法:
(1)按下按鈕的時(shí)候給出一個(gè)<div>提示正在執(zhí)行任務(wù),執(zhí)行完畢讓這個(gè)<div>隱藏
(2)按下按鈕的時(shí)候跳轉(zhuǎn)到一個(gè)提示任務(wù)正在執(zhí)行的頁(yè)面,執(zhí)行完畢了再跳轉(zhuǎn)回來
(3)做一個(gè)任務(wù)類,開啟另外一個(gè)線程執(zhí)行任務(wù),同時(shí)在客戶端或者服務(wù)器端保存這個(gè)類的實(shí)例來跟蹤任務(wù)的執(zhí)行情況(1)和(2)的情況用的比較多,也比較簡(jiǎn)單,缺點(diǎn)是不能實(shí)時(shí)的知道任務(wù)的執(zhí)行進(jìn)度,而且時(shí)間一長(zhǎng)可能會(huì)超時(shí),(3)的方法就會(huì)比較好的解決上面說的2個(gè)缺點(diǎn)。下面著重說一下(3)的實(shí)現(xiàn)方法,先從簡(jiǎn)單開始,我們做一個(gè)任務(wù)類,在客戶端時(shí)時(shí)(暫且刷新時(shí)間為1秒)得知任務(wù)執(zhí)行了多少時(shí)間,并且在成功完成任務(wù)后給出執(zhí)行時(shí)間,在任務(wù)出錯(cuò)的時(shí)候給出出錯(cuò)的時(shí)間。
ASP.NET多線程前臺(tái)
- <form id="Form1" method="post" runat="server">
- <asp:label id="lab_state" runat="server"></asp:label><br>
- <asp:Button id="btn_startwork" runat="server" Text="運(yùn)行一個(gè)長(zhǎng)時(shí)間的任務(wù)"></asp:Button>
- </form>
ASP.NET多線程后臺(tái)
先是一些類的申明:
- protected System.Web.UI.WebControls.Button btn_startwork;
- protected System.Web.UI.WebControls.Label lab_state;
- //前面2個(gè)是vs.net自己生成的
- protected work w;
- 在Page_Load里面輸入以下代碼:
- if(Session["work"]==null)
- {
- w=new work();
- Session["work"]=w;
- }
- else
- {
- w=(work)Session["work"];
- }
- switch(w.State)
- {
- case 0:
- {
- this.lab_state.Text="還沒有開始任務(wù)";
- break;
- }
- case 1:
- {
- this.lab_state.Text="任務(wù)進(jìn)行了"+((TimeSpan)(DateTime.Now-w.StartTime)).TotalSeconds+"秒";
- this.btn_startwork.Enabled=false;
- Page.RegisterStartupScript("","<script>window.setTimeout(’locationlocation.href=location.href’,1000);</script>");
- //不斷的刷新本頁(yè)面,隨時(shí)更新任務(wù)的狀態(tài)
- break;
- }
- case 2:
- {
- this.lab_state.Text="任務(wù)結(jié)束,并且成功執(zhí)行所有操作,用時(shí)"+((TimeSpan)(w.FinishTime-w.StartTime)).TotalSeconds+"秒";
- this.btn_startwork.Enabled=true;
- break;
- }
- case 3:
- {
- this.lab_state.Text="任務(wù)結(jié)束,在"+((TimeSpan)(w.ErrorTime-w.StartTime)).TotalSeconds+"秒的時(shí)候發(fā)生錯(cuò)誤導(dǎo)致任務(wù)失敗";
- this.btn_startwork.Enabled=true;
- break;
- }
- }
在按鈕單擊事件內(nèi)輸入以下代碼:
- if(w.State!=1)
- {
- this.btn_startwork.Enabled=false;
- w.runwork();
- Page.RegisterStartupScript("","<script>locationlocation.href=location.href;</script>");
- //立即刷新頁(yè)面
- }
另外建立一個(gè)任務(wù)類,代碼如下:
- public class work
- {
- public int State=0;//0-沒有開始,1-正在運(yùn)行,2-成功結(jié)束,3-失敗結(jié)束
- public DateTime StartTime;
- public DateTime FinishTime;
- public DateTime ErrorTime;
- public void runwork()
- {
- lock(this)//確保臨界區(qū)被一個(gè)Thread所占用
- {
- if(State!=1)
- {
- State=1;
- StartTime=DateTime.Now;
- System.Threading.Thread thread=new System.Threading.Thread(new System.Threading.ThreadStart(dowork));
- thread.Start();
- }
- }
- }
- private void dowork()
- {
- try
- {
- SqlConnection conn=new SqlConnection(System.Configuration.ConfigurationSettings.AppSettings["conn"]);
- SqlCommand cmd=new SqlCommand("Insert Into test (test)values(’test’)",conn);
- conn.Open();
- for(int i=0;i<5000;i++)cmd.ExecuteNonQuery();
- conn.Close();
- //以上代碼執(zhí)行一個(gè)比較消耗時(shí)間的數(shù)據(jù)庫(kù)操作
- State=2;
- }
- catch
- {
- ErrorTime=DateTime.Now;
- State=3;
- }
- finally
- {
- FinishTime=DateTime.Now;
- }
- }
- }
- }
運(yùn)行這個(gè)頁(yè)面,看到每秒頁(yè)面刷新一次反饋任務(wù)執(zhí)行到現(xiàn)在的時(shí)間,在結(jié)束后給出任務(wù)總的用時(shí)。(如果任務(wù)出錯(cuò)也給出出錯(cuò)時(shí)間)
(這個(gè)ASP.NET多線程示例比較簡(jiǎn)單,基本能實(shí)現(xiàn)長(zhǎng)時(shí)間的任務(wù)執(zhí)行與客戶端的交互,但是界面不是很友善,而且如果有很多項(xiàng)操作的話,只能給出執(zhí)行了多少時(shí)間,不能顯示執(zhí)行到第幾項(xiàng)任務(wù),在下一篇文章中,將會(huì)改進(jìn)這個(gè)類和界面)
【編輯推薦】