|
在asp.net中執(zhí)行一個(gè)長時(shí)間的操作,有的時(shí)候需要在在客戶端有一個(gè)反饋能了解到任務(wù)的執(zhí)行進(jìn)度,大致看了一下有這么幾種做法:
(1)按下按鈕的時(shí)候給出一個(gè)<div>提示正在執(zhí)行任務(wù),執(zhí)行完畢讓這個(gè)<div>隱藏 (2)按下按鈕的時(shí)候跳轉(zhuǎn)到一個(gè)提示任務(wù)正在執(zhí)行的頁面,執(zhí)行完畢了再跳轉(zhuǎn)回來 (3)做一個(gè)任務(wù)類,開啟另外一個(gè)線程執(zhí)行任務(wù),同時(shí)在客戶端或者服務(wù)器端保存這個(gè)類的實(shí)例來跟蹤任務(wù)的執(zhí)行情況 (1)和(2)的情況用的比較多,也比較簡單,缺點(diǎn)是不能實(shí)時(shí)的知道任務(wù)的執(zhí)行進(jìn)度,而且時(shí)間一長可能會(huì)超時(shí),(3)的方法就會(huì)比較好的解決上面說的2個(gè)缺點(diǎn)。下面著重說一下(3)的實(shí)現(xiàn)方法,先從簡單開始,我們做一個(gè)任務(wù)類,在客戶端時(shí)時(shí)(暫且刷新時(shí)間為1秒)得知任務(wù)執(zhí)行了多少時(shí)間,并且在成功完成任務(wù)后給出執(zhí)行時(shí)間,在任務(wù)出錯(cuò)的時(shí)候給出出錯(cuò)的時(shí)間。
前臺(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è)長時(shí)間的任務(wù)"></asp:Button> </form> 后臺(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('location.href=location.href',1000);</script>"); //不斷的刷新本頁面,隨時(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>location.href=location.href;</script>"); //立即刷新頁面 } 另外建立一個(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ù)庫操作 State=2; } catch { ErrorTime=DateTime.Now; State=3; } finally { FinishTime=DateTime.Now; } } } } 運(yùn)行這個(gè)頁面,看到每秒頁面刷新一次反饋任務(wù)執(zhí)行到現(xiàn)在的時(shí)間,在結(jié)束后給出任務(wù)總的用時(shí)。(如果任務(wù)出錯(cuò)也給出出錯(cuò)時(shí)間)
(這個(gè)示例比較簡單,基本能實(shí)現(xiàn)長時(shí)間的任務(wù)執(zhí)行與客戶端的交互,但是界面不是很友善,而且如果有很多項(xiàng)操作的話,只能給出執(zhí)行了多少時(shí)間,不能顯示執(zhí)行到第幾項(xiàng)任務(wù),在下一篇文章中,將會(huì)改進(jìn)這個(gè)類和界面)
|
|
|