在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í)間。
前臺
<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>
后臺
先是一些類的申明:
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è)類和界面)
原文:http://www.cnblogs.com/lovecherry/archive/2005/04/10/135060.html