IsPostBack 的介绍

这涉及到aspx的页面回传机制的基础知识 postback是回传 即页面在首次加载后向服务器提交数据,然后服务器把处理好的数据传递到客户端并显示出来,就叫postback,
ispostback只是一个属性,即判断页面是否是回传,if(!Ispostback)就表示页面是首次加载,这是很常用的一个判断方式.一个页面只能加载一次,但可以在加载后反复postback.

1、IsPostBack介绍
Page.IsPostBack是一个标志:当前请求是否第一次打开。 调用方法为:Page.IsPostBack或者IsPostBack或者this.IsPostBack或者this.Page.IsPostBack,它们都等价。
  1)当通过IE的地址栏等方式打开一个URL时是第一次打开, 当通过页面的提交按钮或能
引起提交的按钮以POST的方式提交的服务器时,页面就不再是第一次打开了。(每点击一次按钮,都是一次加载)
  2)IsPostBack只有在第一次打开的时候是false,其它时候都是true
  3).Net判断一个Page是否第一次打开的方法:Request.Form.Count>0
  4)每次页面Load的时候,根据需要把每次都要加载的代码放在IsPostBack中,只需要
加载一次的代码放在if(!IsPostBack)中。
  5)每次用户回传服务器任何信息的时候,都会引发isPostBack属性用来判断此用户是

否曾经做过登陆或者其他事件
  6 if(!IsPostBack)
  { 
  Response.Write("第一次提交!"); 
  } 
  if(IsPostBack) 
  { 
  Response.Write("按按钮!"); 
  } 
2 IsPostBack结论

结论① 对于使用Server.Transfer进行迁移时迁移到的页面其IsPostBack=false。

(迁移到的页面第一次打开)
结论② Post方式如果Request中没有请求值,即Request.Form =null则IsPostBack=false
;Get方式如果Request中没有请求值,即Request.QueryString =null则IsPostBack=false。
结论③ 如果QueryString或Form虽然有请求值,但是QueryString或Form中的Key没有
“__VIEWSTATE”和“__EVENTTARGET”和“__VIEWSTATEFIELDCOUNT”,并且没有键为“null”,值以“__VIEWSTATE”开头并且也没有值为“__EVENTTARGET”的键值对,则IsPostBack=false。
结论④ 使用Response.Redirect方式向自画面迁移时,此时IsPostBack=false。
结论⑤ 发生跨页提交(CrossPagePostBack),当访问PreviousPage属性的时候,对于
源PageIsPostBack=true。

结论⑥ 发生跨页提交(CrossPagePostBack)时目标页面是IsPostBack=false
结论⑦ 使用Server.Execute迁移到的页面其IsPostBack=false。
结论⑧ 在Page运行期间其对应的DLL被更新了并且Page的树结构发生过变化,这种情况
下请求时IsPostBack=false。
可以这样来理解这些结论:一般情况判断Request中如果没有请求值则IsPostBack=false
。如果有请求值但是不包括“__VIEWSTATE”等一些特殊的键或值,则IsPostBack=false(每次请求后.Net框架会将一些特殊的隐藏域 “__VIEWSTATE”等返回给客户端)。还有一些特殊的情形是上面的规则不能正确判断的需要特殊处理的,这些情形包括Server.Transfer,Response.Redirect,CrossPagePostBack,Server.Execute,发
生了页面元素变化及重新编译。

 

 

 

 

获取一个值,该值指示该页是否正为响应客户端回发而加载,或者它是否正被首次加载和访问。

if(!this.IsPostBack) {

} 通常用在page_load中,获取一个值,该值指示该页是否正为响应客户端回发而加载,或者它是否正被首次加载和访问,如果是为响应客户端回发而加载该页,则为true;否则为 false。 比如如果你在页面上按了一下按钮,那么在Page_load时,你就会看到isPostBack等于true;如果你的页面是第一次被打开,则返回false; !this.IsPostBack就是一直取true,把这个页面里面的东西一直当成第一次打开

 

!IsPostBack==ture 当前页面是第一次加载 IsPostBack 由于用户交互(page.submit())提交页面而产生的加载,二次加载 isPostback属性获取一个值,该值指示该页是否正为响应客户端回发而加载,或者它是否正被首次加载和访问。如果是为响应客户端回发而加载该页,则为true;否则为 false(第一次加载)所以if(!IsPostBack)的意思就是如果不是了响应客户端回发而加载该页的时候执行{}里的数据---即第一次加载。IsPostBack的意思就是第二次(IsPostBack本身有回传信息),!IsPostBack的意思就是不是第二次---即第一次。所以你的以上代码最好放在里面,这样在你使用分页的时候就知道为什么了,你可以为你这datagird加入分页,然后分别把代码放在{}里面和外面看看效果,只有为第一次加载时执行此代码。

一个页面第一次显示的时候 isPostBack=false 然后你在这个页面上点击按钮或其它东西提交的时候, isPostBack=true 一般这个函数里面的内容是指第一次打开这个页面的时候要做的事. 一般做一些初始化什么的东西.

.假设你在page_load()里定义textbox为空,在button_click事件里给写一段往数据库存textbox的值.当用户往 textbox里输入值以后,点击button时,就应该可以把往textbox里输入的值存到数据库里.可实际上,当点击button的时候,系统会自动先重新执行page_load(),也就是把柄textbox清空,再执行click 事件,你往数据库里存的值永远都是空.所以要在page_Load() 里加入if(!IsPostBack).

答2: 是第一次加载页面时Page.ispostback是false,当触发该页面的web控件时,产生回发,这时候 该属性会是true,所以要用if(!IsPostback)来判断,只在第一次进入页面时加载数据。 ______________________________________________________________________________________________ 答 3: 总之一句话,放在使page_load里的代码只执行一遍 ______________________________________________________________________________________________ 答 4: 第一次显示的时候 IFIsPostBack)这个IF里面的语句是执行的

以后页面重新刷新的话 这个IF里面的语句不执行

______________________________________________________________________________________________ 答 5: 首次建立页面 IsPostBack为false 用来控制 里边的语句只执行一次 ______________________________________________________________________________________________ 答 6: Page.IsPostBack 属性 获取一个值,该值指示该页是否正为响应客户端回发而加载,或者它是否正被首次加载和访问。 如果是为响应客户端回发而加载该页,则为 true;否则为 false。 if(!IsPostBack) { .... } 表示该页面首次加载时需要做哪些事情

____________________________

 

1、  IsPostBack是Page类有一个bool类型的属性,用来判断针对当前Form的请求是第一次还是非第一次请求。当IsPostBack=true时表示非第一次请求,我们称为PostBack,当IsPostBack=false时表示第一次请求。在asp.net框架内部有很多的场景需要判断IsPostBack,比如   LoadAllState等操作就需要在PostBack的时候进行。对于我们自己使用WebForm进行开发时,经常会在Page_Load中对IsPostBack进行判断,因  为第一次请求的时候会执行Page_Load,在非第一次请求的时候也会执行Page_Load。为什么对同一个Form有多次请求呢?asp.net中引入了服务  器端事件,支持服务器端事件的控件,会发出对当前Form的请求,这样在很多情形下我们就需要区别是否是对这个Form的第一次请求。      if(!IsPostBack)的使用   获取一个值,该值指示该页是否正为响应客户端回发而加载,或者它是否正被首次加载和访问。 if(!this.IsPostBack) { }  通常用在page_load中,获取一个值,该值指示该页是否正为响应客户端回发而加载,或者它是否正被首次加载和访问,如果是为响应客户端回发而加载该页,则为true;否则为 false(第一次)。  比如如果你在页面上按了一下按钮,那么在Page_load时,你就会看到isPostBack等于true;如果你的页面是第一次被打开,则返回false; !this.IsPostBack就是一直取true,把这个页面里面的东西一直当成第一次打开    !IsPostBack==ture 当前页面是第一次加载  IsPostBack 由于用户交互(page.submit())提交页面而产生的加载,二次加载 isPostback属性获取一个值,该值指示该页是否正为响应客户端回发而加载,或者它是否正被首次加载和访问。如果是为响应客户端回发而加载该页,则为true;否则为 false(第一次加载)。 

 

所以if(!IsPostBack)的意思就是如果不是了响应客户端回发而加载该页的时候执行{}里的数据---即第一次加载。IsPostBack的意思就是第二次 (IsPostBack本身有回传信息),!IsPostBack的意思就是不是第二次---即第一次。所以你的以上代码最好放在里面,这样在你使用分页的时候就知道为什么了,你可以为你这datagird加入分页,然后分别把代码放在{}里面和外面看看效果,只有为第一次加载时执行此代码。 一个页面第一次显示的时候 isPostBack=false  然后你在这个页面上点击按钮或其它东西提交的时候, isPostBack=true  一般这个函数里面的内容是指第一次打开这个页面的时候要做的事. 一般做一些初始化什么的东西.  .假设你在page_load()里定义textbox为空,在button_click事件里给写一段往数据库存textbox的值.当用户往 textbox里输入值以后,点击button时,就应该可以把往textbox里输入的值存到数据库里.可实际上,当点击button的时候,系统会自动先重新执行page_load(),也就是把柄textbox清空,再执行click 事件,你往数据库里存的值永远都是空.所以要在page_Load() 里加入if(!IsPostBack). 

 

3  IsPostBack推论过程

 

下面是根据.Net框架中的源代码,来分析IsPostBack是如何判断出来的。对于这些结论的推断本人做了相关的试验来证明推论的正确性,由于篇幅的原因没有将这些试验代码体现出来。另外不可能将全部的.Net框架的代码都体现出来,只是将相关的代码片段列出,说明推断的依据。另外由于本人水平有限对.Net框架的代码理解还存在的不足的地方,请发现后进行指正,谢谢。

 

publicbool IsPostBack
{
  get
  {
    if (this._requestValueCollection == null)
    {
      return false;
    }
    if (this._isCrossPagePostBack)
    {
      return true;
    }
    if (this_pageFlags[8])
    {
      return false;
    }
  return (
(
(this.Context.ServerExecuteDepth <= 0) ||
((this.Context.Handler != null) &&
(base.GetType() == this.Context.Handler.GetType()))
) && !this._fPageLayoutChanged
);
  }
}

 

 

 

我们将每一个if判断作为一个小节,作如下的分析。

 

 

 

3.1 this._requestValueCollection == null

 

 

 

if (this._requestValueCollection == null)
{
      return false;
}

 

 

 

可以看出_requestValueCollection等于null时IsPostBack就等于false。

 

 

 

在Page.ProcessRequestMain(bool, bool)中有如下的代码:

 

 

 

if (this.PageAdapter != null)
{
  this._requestValueCollection = this.PageAdapter.DeterminePostBackMode();
}
else
{
  this._requestValueCollection = this.DeterminePostBackMode();
}

 

PageAdapter.DeterminePostBackMode最终还是调用了Page.DeterminePostBackMode,下面我们看Page.DeterminePostBackMode如何实现。

 

protected internal virtual NameValueCollection DeterminePostBackMode()
{
  if (this.Context.Request == null)
  {
    return null;
  }
  if (this.Context.PreventPostback)
  {
    return null;
  }
  NameValueCollection collectionBasedOnMethod = this.GetCollectionBasedOnMethod(false);
  if (collectionBasedOnMethod == null)
  {
    return null;
  }
  bool flag = false;
  string[] values = collectionBasedOnMethod.GetValues((string) null);
  if (values != null)
  {
    int length = values.Length;
    for (int i = 0; i < length; i++)
    {
      if (values[i].StartsWith("__VIEWSTATE", StringComparison.Ordinal) ||
(values[i] == "__EVENTTARGET"))
      {
        flag = true;
        break;
      }
    }
  }
if (((collectionBasedOnMethod["__VIEWSTATE"] == null) && (collectionBasedOnMethod["__VIEWSTATEFIELDCOUNT"] == null)) && ((collectionBasedOnMethod["__EVENTTARGET"] == null) && !flag))
  {
    return null;
  }
if (this.Request.QueryStringText.IndexOf(
HttpResponse.RedirectQueryStringAssignment, StringComparison.Ordinal) != -1)
  {
    collectionBasedOnMethod = null;
  }
  return collectionBasedOnMethod;
}

 

这个函数中返回null就意味者IsPostBack=false,将上面函数中每个返回为null的地方作如下的分析。

 

 

3.1.1 this.Context.Request == null

 

 

 

  if (this.Context.Request == null)
  {
    return null;
  }

 

 

 

this.Context.Request == null应该只有在异常的情况下会发生,正常情况下会在HttpRuntime.ProcessRequestInternal中创建HttpContext及HttpRequest对象。

 

 

3.1.2 this.Context.PreventPostback

 

 

 

  if (this.Context.PreventPostback)
  {
    return null;
  }

 

 

 

在HttpServerUtility.Transfer中会使用PreventPostback,其代码如下:

 

 

 

    public void Transfer(string path)
    {
      bool preventPostback = this._context.PreventPostback;
      this._context.PreventPostback = true;
      this.Transfer(path, true);
      this._context.PreventPostback = preventPostback;
    }

 

 

 

在调用Server.Transfer进行画面迁移时设置Context.PreventPostback=ture。此处得出结论①:对于使用Server.Transfer进行迁移时迁移到的页面其IsPostBack=false。

 

 

 

3.1.3  collectionBasedOnMethod == null

 

 

 

  NameValueCollection collectionBasedOnMethod = this.GetCollectionBasedOnMethod(false);
  if (collectionBasedOnMethod == null)
  {
    return null;
  }

 

 

 

调用了Page.GetCollectionBasedOnMethod后其返回值进行判断。如果其返回值为null则IsPostBack为false。Page.GetCollectionBasedOnMethod的定义如下:

 

 

 

    internal NameValueCollection GetCollectionBasedOnMethod(bool dontReturnNull)
    {
      if (this._request.HttpVerb == HttpVerb.POST)
      {
        if (!dontReturnNull && !this._request.HasForm)
        {
          return null;
        }
        return this._request.Form;
      }
      if (!dontReturnNull && !this._request.HasQueryString)
      {
        return null;
      }
      return this._request.QueryString;
    }

 

 

 

从上面的代码可以看出返回值为null的情形是_request.HasForm=null或_request.HasQueryString=null。此处得出结论②:Post方式如果Request中没有请求值,即Request.Form =null则IsPostBack=false;Get方式如果Request中没有请求值,即Request.QueryString =null则IsPostBack=false。

 

 

 

 

 

 

 

3.1.4 ((collectionBasedOnMethod["__VIEWSTATE"] == null) && (collectionBasedOnMethod["__VIEWSTATEFIELDCOUNT"] == null)) && ((collectionBasedOnMethod["__EVENTTARGET"] == null) && !flag)

 

 

 

bool flag = false;
  string[] values = collectionBasedOnMethod.GetValues((string) null);
  if (values != null)
  {
    int length = values.Length;
    for (int i = 0; i < length; i++)
    {
      if (values[i].StartsWith("__VIEWSTATE", StringComparison.Ordinal) ||
(values[i] == "__EVENTTARGET"))
      {
        flag = true;
        break;
      }
    }
  }

 

 

 

上面这段代码的意思是判断请求的键值对中是否存在没有键,其值以“__VIEWSTATE”开头或者其值为“__EVENTTARGET”。例如如下的Get请求方式会使得flag=true。

 

 

 

…/defalt.aspx?__VIEWSTATE

 

 

 

…/defalt.aspx?__EVENTTARGET

 

 

 

对于Get方式“?__VIEWSTATE=”会将__VIEWSTATE作为请求的键,其值为“”,但是“?__VIEWSTATE”会认为其键为“null”,其值为“__VIEWSTATE”

 

 

 

if (
((collectionBasedOnMethod["__VIEWSTATE"] == null) && (collectionBasedOnMethod["__VIEWSTATEFIELDCOUNT"] == null)) && ((collectionBasedOnMethod["__EVENTTARGET"] == null) && !flag))
{
return null;
}

 

 

 

如上的条件意味着请求的键中同时没有“__VIEWSTATE”,“__EVENTTARGET”,“__VIEWSTATEFIELDCOUNT”,并且flag为false则返回null。flag为false意味着没有键为“null”值以“__VIEWSTATE”开头并且也没有值为“__EVENTTARGET”的键值对。

 

 

 

此处得出结论③如果QueryString或Form虽然有请求值,但是QueryString或Form中的Key没有“__VIEWSTATE”和“__EVENTTARGET”和“__VIEWSTATEFIELDCOUNT”,并且没有键为“null”值以“__VIEWSTATE”开头并且也没有值为“__EVENTTARGET”的键值对,则IsPostBack=false。

 

 

 

 

 

 

 

3.1.5 this.Request.QueryStringText.IndexOf(HttpResponse.RedirectQueryStringAssignment, StringComparison.Ordinal) != -1

 

 

 

if (this.Request.QueryStringText.IndexOf(HttpResponse.RedirectQueryStringAssignment, StringComparison.Ordinal) != -1)
{
collectionBasedOnMethod = null;
}

 

 

 

HttpResponse.RedirectQueryStringAssignment的值为“__redir=1”,上面的代码的意思是如果QueryStringText中包括包括“__redir=1”则返回null。在HttpRequest.Redirect中会判断如果IsPostBack为true,并且URL中不包含有“__redir=1”时,会给URL中增加“__redir=1”。一般情况下我们使用request.Redirect迁移到的页面都应该是IsPostBack=false,有一种特殊的情形是使用request.Redirect迁移到当前页,此时IsPostBack为true。此种情况发生时在request.Redirect中给URL中增加“__redir=1”。执行到page. ProcessRequestMain时会重新将IsPostBack判断为fales。

 

 

 

此处得出结论④使用Response.Redirect方式向自画面迁移时,此时IsPostBack=false。

 

 

 

此时大家可能会有疑问为什么使用Response.Redirect方式向自画面迁移时要特殊处理,使用Response.Redirect向其他画面迁移为什么不要。使用Response.Redirect向其他画面迁移时Response.Form=null,Response.QueryString=null,所以可以判断是IsPostBack=false。但是使用Response.Redirect方式向自画面迁移时Response.QueryString<>null,所以要特殊判断。

 

 

 

3.2 this._isCrossPagePostBack

 

 

 

    if (this._isCrossPagePostBack)
    {
      return true;
    }

 

 

 

在Page的PreviousPage属性中会对_isCrossPagePostBack进行设置,具体代码如下:

 

 

 

public Page PreviousPage
{
  get
  {
      …
ITypedWebObjectFactory vPathBuildResult = (ITypedWebObjectFactory) BuildManager.GetVPathBuildResult(this.Context, this._previousPagePath);
      if (typeof(Page).IsAssignableFrom(vPathBuildResult.InstantiatedType))
      {
        this._previousPage = (Page) vPathBuildResult.CreateInstance();
        this._previousPage._isCrossPagePostBack = true;
        this.Server.Execute(this._previousPage, TextWriter.Null, true, false);
      }
    }
    return this._previousPage;
  }
}

 

 

 

在发生跨页面提交的时候,当访问PreviousPage属性的时候源Page的IsCrossPagePostBack会被设置true。此处得出结论⑤发生跨页提交(CrossPagePostBack),当访问PreviousPage属性的时候,对于源Page,IsPostBack=true。

 

 

 

 

 

 

 

3.3 this._pageFlags[8]

 

 

 

if (this._pageFlags[8]) { return false; }

 

 

 

在Page. ProcessRequestMain中有如下的代码片断对_pageFlags[8]进行赋值。

 

 

 

else if (!this.IsCrossPagePostBack) {     VirtualPath path = null;     if (this._requestValueCollection["__PREVIOUSPAGE"] != null)     {         try         {             path = VirtualPath.CreateNonRelativeAllowNull( DecryptString(this._requestValueCollection["__PREVIOUSPAGE"]));         }         catch (CryptographicException)         {             this._pageFlags[8] = true;         }         if ((path != null) && (path != this.Request.CurrentExecutionFilePathObject))         {             this._pageFlags[8] = true;             this._previousPagePath = path;         }     } }

 

 

 

解密发生异常时_pageFlags[8]为true这种异常发生的可能性比较小我们忽略,重点看另外一种情形,将这种情形的所有条件结合起来就是IsCrossPagePostBack=false && _requestValueCollection["__PREVIOUSPAGE"] != null && path != null && (path != this.Request.CurrentExecutionFilePathObject)。发生跨页提交时对于目标页面IsCrossPagePostBack=false,此时源页面的"__PREVIOUSPAGE"等信息会提交给目标页面,所以_requestValueCollection["__PREVIOUSPAGE"] != null。此时当前请求的CurrentExecutionFilePathObject是根据目标页的路径生成的,与使用_requestValueCollection["__PREVIOUSPAGE"]生成的path对象不同。

 

 

 

此处得出结论⑥发生跨页提交(CrossPagePostBack)时目标页面是IsPostBack=false。为什么需要对CrossPagePostBack的目标页面做这样的处理呢?发生CrossPagePostBack时,会将源页面的信息提交给目标页面此时Request.Form!=null,而且包括__VIEWSTATE等键按照其他的规则会判断为IsPostBack=true,所以需要对CrossPagePostBack的目标页面做特殊的判断。

 

 

 

 

 

 

 

3.4 (this.Context.ServerExecuteDepth <= 0) || ((this.Context.Handler != null) && (base.GetType() == this.Context.Handler.GetType()))

 

 

 

在HttpServerUtility中有如下的代码对Context. ServerExecuteDepth进行了操作。

 

 

 

public void Execute(string path, TextWriter writer, bool preserveForm) { … try { this._context.ServerExecuteDepth++; handler = this._context.ApplicationInstance.MapHttpHandler(this._context, request.RequestType, path3, filename, useAppConfig); } finally { this._context.ServerExecuteDepth--; } … }

 

 

 

在HttpServerUtility.ExecuteInternal中也有一处对Context.ServerExecuteDepth类似的操作。HttpServerUtility.Execute会调用HttpServerUtility.ExecuteInternal。从此可以看出Context.ServerExecuteDepth是表示Server.Execute中的执行深度。在调用Server.Execute时Context.ServerExecuteDepth>0。另外调用Server.Execute后Context.Handle中存储的还是原来的页对象,也就是说base.GetType()!= this.Context.Handler.GetType()。这样对于Server.Execute来说this.Context.ServerExecuteDepth <= 0) || ((this.Context.Handler != null)这个条件为false。此处得出结论⑦使用Server.Execute迁移到的页面其IsPostBack=false。此处我们会有疑问,为什么需要对Server.Execute进行特殊的判断呢?理由是使用Server.Execute时会将源Page中的隐含域提交,此时Request.Form!=null,而且包括__VIEWSTATE等键按照其他的规则会判断为IsPostBack=true。

 

 

 

3.5 this._fPageLayoutChanged

 

 

 

fPageLayoutChanged从这个变量的字面意思来看是Page的Layout发生了变化。

 

 

 

在Page.LaodAllState中代码片断如下:

 

 

 

private void LoadAllState() { … string s = (string) second.First; int num = int.Parse(s, NumberFormatInfo.InvariantInfo); this._fPageLayoutChanged = num != this.GetTypeHashCode(); … }

 

 

 

其意思是现在得到的HashCode和存储在ViewState中的HashCode不一致时fPageLayoutChanged=true。GetTypeHashCode()会返回一个HashCode,而且这个方法是对aspx进行编译的时候产生的,只有在页面上的元素发生了变化的时候其返回的值会发生变化。此处得出结论⑧在Page运行期间其对应的DLL被更新了并且Page的树结构发生过变化,这种情况下请求时IsPostBack=false。

 

 

posted @ 2013-10-16 13:45  Sally Yu  阅读(313)  评论(0编辑  收藏  举报