IsPostBack深入探讨

简介:

1         IsPostBack介绍

IsPostBackPage类有一个bool类型的属性,用来判断针对当前Form的请求是第一次还是非第一次请求。当IsPostBacktrue时表示非第一次请求,我们称为PostBack,当IsPostBackfalse时表示第一次请求。在asp.net框架内部有很多的场景需要判断IsPostBack,比如LoadAllState等操作就需要在PostBack的时候进行。对于我们自己使用WebForm进行开发时,经常会在Page_Load中对IsPostBack进行判断,因为第一次请求的时候会执行Page_Load,在非第一次请求的时候也会执行Page_Load。为什么对同一个Form有多次请求呢?asp.net中引入了服务器端事件,支持服务器端事件的控件,会发出对当前Form的请求,这样在很多情形下我们就需要区别是否是对这个Form的第一次请求。

2         IsPostBack结论

本人对.Net的源代码中相关的处理进行的分析得到如下的结论:

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

结论②    Post方式如果Request中没有请求值,即Request.Form =nullIsPostBackfalseGet方式如果Request中没有请求值,即Request.QueryString =nullIsPostBackfalse

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

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

结论⑤    发生跨页提交(CrossPagePostBack),当访问PreviousPage属性的时候,对于源PageIsPostBack=true

结论⑥    发生跨页提交(CrossPagePostBack)时目标页面是IsPostBackfalse

结论⑦    使用Server.Execute迁移到的页面其IsPostBackfalse

结论⑧    Page运行期间其对应的DLL被更新了并且Page的树结构发生过变化,这种情况下请求时IsPostBackfalse

可以这样来理解这些结论:一般情况判断Request中如果没有请求值则IsPostBackfalse。如果有请求值但是不包括“__VIEWSTATE”等一些特殊的键或值,则IsPostBackfalse(每次请求后.Net框架会将一些特殊的隐藏域“__VIEWSTATE”等返回给客户端)。还有一些特殊的情形是上面的规则不能正确判断的需要特殊处理的,这些情形包括Server.TransferResponse.RedirectCrossPagePostBackServer.Execute,发生了页面元素变化及重新编译。

一般来说记住上面的结论就可以,如果您有兴趣,或者怀疑请继续看下面的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等于nullIsPostBack就等于false

Page.ProcessRequestMain(boolbool)中有如下的代码:

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((stringnull);

    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就意味者IsPostBackfalse,将上面函数中每个返回为null的地方作如下的分析。

3.1.1        this.Context.Request == null

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

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

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.PreventPostbackture。此处得出结论①:对于使用Server.Transfer进行迁移时迁移到的页面其IsPostBackfalse

3.1.3        collectionBasedOnMethod == null

    NameValueCollection collectionBasedOnMethod = this.GetCollectionBasedOnMethod(false);

    if (collectionBasedOnMethod == null)

    {

        return null;

    }

调用了Page.GetCollectionBasedOnMethod后其返回值进行判断。如果其返回值为nullIsPostBackfalsePage.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.HasFormnull_request.HasQueryStringnull。此处得出结论②:Post方式如果Request中没有请求值,即Request.Form =nullIsPostBackfalseGet方式如果Request中没有请求值,即Request.QueryString =nullIsPostBackfalse

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

    bool flag = false;

    string[] values = collectionBasedOnMethod.GetValues((stringnull);

    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, truefalse);

            }

        }

        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.ExecuteContext.ServerExecuteDepth>0。另外调用Server.ExecuteContext.Handle中存储的还是原来的页对象,也就是说base.GetType()!= this.Context.Handler.GetType()。这样对于Server.Execute来说this.Context.ServerExecuteDepth <= 0) || ((this.Context.Handler != null)这个条件为false此处得出结论⑦使用Server.Execute迁移到的页面其IsPostBackfalse。此处我们会有疑问,为什么需要对Server.Execute进行特殊的判断呢?理由是使用Server.Execute时会将源Page中的隐含域提交,此时Request.Form!=null,而且包括__VIEWSTATE等键按照其他的规则会判断为IsPostBacktrue

3.5        this._fPageLayoutChanged

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

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 的树结构发生过变化,这种情况下请求时IsPostBackfalse



本文转自94cool博客园博客,原文链接:http://www.cnblogs.com/94cool/articles/1532608.html,如需转载请自行联系原作者

相关文章
|
3月前
|
设计模式 人工智能 自然语言处理
【利用AI让知识体系化】简要了解面向对象编程设计(二)
【利用AI让知识体系化】简要了解面向对象编程设计
|
3月前
|
设计模式 人工智能 关系型数据库
【利用AI让知识体系化】简要了解面向对象编程设计(一)
【利用AI让知识体系化】简要了解面向对象编程设计
|
6月前
|
自然语言处理 架构师
技术人修炼之道阅读笔记(二)重新定义自己
技术人修炼之道阅读笔记(二)重新定义自己
|
12天前
|
监控 前端开发 算法
深入探讨前端性能优化策略
在当今互联网发展日新月异的时代,前端性能优化已成为开发者关注的重要议题。本文将从前端加载优化、代码精简、资源压缩、图像优化等方面深入探讨前端性能优化的策略,为开发者提供实用的指导和建议。
|
3月前
|
Java jenkins 测试技术
深入探讨软件测试技术:方法、工具与最佳实践
深入探讨软件测试技术:方法、工具与最佳实践
65 0
|
4月前
|
机器学习/深度学习 自然语言处理
ChatGPT技术基石之Transformer技术的简介(简单易懂)
ChatGPT技术基石之Transformer技术的简介(简单易懂)
49 0
|
4月前
|
数据挖掘 项目管理
技术写作及技术作者的概述和重要性 - 了解技术写
技术写作是指用简单易懂的语言向特定受众解释复杂概念的一种写作形式。这种写作形式通常用于工程、计算机硬件和软件、金融、消费电子和生物技术等领域。技术作者的主要目标是简化复杂信息并以清晰简洁的方式呈现。技术作者的职责可能包括创建操作指南、用户手册、常见问题解答页面、期刊论文和其他技术内容,以帮助用户理解。最终目标是使用户能够轻松理解和掌握新产品或概念。
90 0
|
6月前
|
机器学习/深度学习 人工智能 算法
这篇科普让你Get所有大模型的基础核心知识点
本文介绍了AI大模型的概念和发展历程。AI大模型是指具有1亿以上参数的机器学习模型,通过在大规模数据集上进行预训练,可以直接支撑各类应用。大模型的发展经历了从萌芽期到AI1.0时期,再到AI2.0时期的飞跃,目前最新发布的大模型参数已经达到了千亿甚至万亿级别。国内外的公司都在积极研发和应用大模型,如OpenAI、Google、Facebook、Microsoft等。国内也有百度、阿里巴巴、万维、商汤科技等公司发布了自己的大模型产品。大模型的建造离不开算力资源、算法人才、数据积累等核心要素。此外,文章还列举了一些与大模型相关的专业名词,如算法、模型参数、训练数据、Token等。
|
SQL 设计模式 容灾
如何通过项目更好的学习并发编程?
秒杀系统最大的几个挑战就是:巨大的瞬时流量,热点数据,刷子流量,商品超买和库存超卖问题,这次冰河会带你从方案、架构和代码等多个方面解决这些核心问题。跟冰河一起开启一段新的技术旅程,在接下来的一段时间内,我要手把手教你搭建并研发一套抗瞬时百万流量的秒杀系统,真正贴近大厂核心项目,为你的职业生涯保驾护航
123 0
如何通过项目更好的学习并发编程?
|
算法 搜索推荐 测试技术
9个数据科学中常见距离度量总结以及优缺点概述(上)
9个数据科学中常见距离度量总结以及优缺点概述
284 0
9个数据科学中常见距离度量总结以及优缺点概述(上)