前章

前段时间碰到button控件的一个属性问题UseSubmitBehavior。

在MSDN上查看,UseSubmitBehavior 属性时,有这样一句话
“获取或设置一个布尔值,该值指示 Button 控件使用客户端浏览器的提交机制还是 ASP.NET 回发机制。 ”

" 默认情况下,此属性的值为 true ,从而导致 Button 控件使用浏览器的提交机制。如果指定为 false ,则 ASP.NET 页框架将客户端脚本添加到页面,以将窗体发送到服务器。"
这里讲到 客户端浏览器的提交机制 和 ASP.NET 回发机制 ,这两种机制有些不明白。于是在论坛上发了个帖子,只有一个人简单的回复了下。讲的内容也只是先前我再msdn上看的介绍UseSubmitBehavior的内容。

可能是问题没有问清楚,于是google了下。发现了一篇文章http://blog.csdn.net/W3031213101/archive/2007/12/18/1946769.aspx 内容如下:

现在就分段进行分析下:

  页面回传与js调用服务端事件、PostBack的原理

第一章、  Asp.net 中服务端控件事件是如何触发的

Asp.net 中在客户端触发服务器端事件分为两种情况:
一.   WebControls 中的 Button 和 HtmlControls 中的 Type 为 submit 的 HtmlInputButton
这两种按钮最终到客户端的表现形式为: < input type ="submit" value ="Submit"> ,这是 Form 表单的提交按钮,点击以后会作为参数发送到服务端,参数是这样的:
控件的 name 属性 = 控件的 value 值,对应上面的例子就是: Submit1= Submit 。服务器端会根据接收到的控件的 name 属性的这个 key 来得知是这个按钮被点击了,从而在服务端触发这个按钮的点击事件。
:这里讲到两种服务器控件<asp:button/>和<asp:imagebutton/>,他们默认 触发事件的方式就如同前面MSDN讲到的浏览器的提交机制。我们可以做个测试查看下,到底是怎么回事。这个很常见,往往最习以为常的东西却是最让人忽视的。
例1:
前台
<asp:Button ID="Button4" runat="server" Text="Button" οnclick="Button4_Click" />
        <asp:ImageButton ID="ImageButton1" runat="server"
            οnclick="ImageButton1_Click" />
后台:
protected void ImageButton1_Click(object sender, ImageClickEventArgs e)
        {
           response.write("1");
        }
        protected void Button4_Click(object sender, EventArgs e)
        {
          response.write("2");
        }
在第一次请求页面以后查看页面源代码,

发现没有什么可以引发事件触发的代码,代码如下:

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" >
<head><title>
无标题页
</title></head>
<body>
<form name="form1" method="post" action="default.aspx" id="form1">
<div>
<input type="hidden" name="__EVENTTARGET" id="__EVENTTARGET" value="" />
<input type="hidden" name="__EVENTARGUMENT" id="__EVENTARGUMENT" value="" />
<input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE" value="/wEPDwUKMTEzODk5NzQ4MWQYAQUeX19Db250cm9sc1JlcXVpcmVQb3N0QmFja0tleV9fFgEFDEltYWdlQnV0dG9uMZP+1UiluyDtaGTdzOnSzX5ckWf7" />
</div>
<script type="text/javascript">
//<![CDATA[
var theForm = document.forms['form1'];
if (!theForm) {
theForm = document.form1;
}
function __doPostBack(eventTarget, eventArgument) {
if (!theForm.onsubmit || (theForm.onsubmit() != false)) {
theForm.__EVENTTARGET.value = eventTarget;
theForm.__EVENTARGUMENT.value = eventArgument;
theForm.submit();
}
}
//]]>
</script>
<div>
<input type="hidden" name="__EVENTVALIDATION" id="__EVENTVALIDATION" value="/wEWAwK53pImAoznisYGAtLCmdMIz6upEG93aRH4h08lrQNs2kst81M=" />
</div>
<h2>正在建设中....</h2>
<input type="button" name="Button1" value="Button" οnclick="javascript:__doPostBack('Button1','')" id="Button1" />
<input type="image" name="ImageButton1" id="ImageButton1" src="" style="border-width:0px;" />
</form>
</body>
</html>
 但是当我们点击button 和imagebutton的时候,事件方法会触发,输出1,或2。我想这就是客户端浏览器的提交机制。
二.   HtmlControls 中的 Type 为 button 的 HtmlInputButton 和其它所有的控件事件,比如 LinkButton 点击, TextBox 的 Change 事件等等:
这些事件在客户端产生后会经过一个统一的机制发送到服务端。
1. 首先 asp.net 页框架会使用两个 Hidden 域来存放表示是哪个控件触发的事件,以及事件的参数:
<!― 表示触发事件的控件,一般是这个控件的 name -->
< input type ="hidden" value ="" />
<!― 表示触发事件的参数 , 一般是当某个控件有两个以上的事件时,用来区别是哪个事件 -->
< input type ="hidden" value ="" />
 
2. 服务端会生成一个 jscript 的方法来处理所有这些事件的发送,这段代码是:
< script language ="javascript" type ="text/javascript">
    function __doPostBack(eventTarget, eventArgument)
    {
        var theform = document.WebForm2;
        theform.__EVENTTARGET.value = eventTarget;
        theform.__EVENTARGUMENT.value = eventArgument;
        theform.submit();
    }
</ script >   
3. 每个会引发服务端事件的控件都会在响应的客户端事件中调用上面的代码:
比如, HtmlControls 中的 Type 为 button 的 HtmlInputButton 的点击事件
<!― 客户端的点击事件调用 __doPostBack , eventTarget 参数为 'Button2' ,表示是 name 为 'Button2’ 控件触发的事件, eventArgument 为空,表示这个 Type 为 button 的 HtmlInputButton 只有一个客户端触发的服务端事件 -->
<input language="javascript" Button2','')" type="button"   value="Button" />
又比如, TextBox 控件的 Change 事件
<!― 客户端的 onchange 事件调用 __doPostBack , eventTarget 参数为 ’TextBox1’ ,表示是 name 为 ’TextBox1’ 控件触发的事件,而 TextBox 控件只有一个客户端触发的服务端事件 TextChanged ,故服务器就会去触发这个 TextBox 的 TextChanged 事件 ->
<input type="text" TextBox1','')" language="javascript" />
4. 客户端触发事件后调用 __doPostBack 方法,将表示触发的控件源的 eventTarget 和事件参数 eventArgument 分别付给两个隐藏域 __EVENTTARGET 和 __EVENTARGUMENT ,然后提交 Form ,在服务端根据 __EVENTTARGET 和 __EVENTARGUMENT 来判断是哪个控件的什么事件触发了。
:以上4个描述讲的大概意思就是,除了第一讲的服务器控件以外,其他的事件的触发都是通过__doPostBack(eventTarget, eventArgument)来解决的。然后我从工具箱里托了一些控件出来,第一个放的是<asp:LinkButton/>,运行查看源代码发现了
function __doPostBack(eventTarget, eventArgument){....}内容存在.。觉得很正确。又拖放了个<asp:TextBox/>控件,运行查看源代码发现并没有 __doPostBack(){}方法,心里就有点怀疑上面的说法。接着拖放了<asp:DropdownList/>,<asp:CheckBox/>,等等几个服务器控件,并没有发现 __doPostBack()。是不是事件没有加,于是在 <asp:TextBox/>加了TextChanged()事件,发现还是老样子。别的几个控件也对应的加上了对应的事件。可也是同样的结果。调试进去后发现,而且事件方法也并没有触发。又开始怀疑是不是上面的错了,最后想了想服务器控件想要回发,不是还要设置AutoPostBack ="true"嘛,于是将此属性添加上去,果然看到了 <asp:LinkButton/>同样的效果,接着我又把对应的事件方法又去掉,只留下 AutoPostBack ="true"。 __doPostBack()依然存在。原来autoposback="true"才是关键.那为什么 <asp:LinkButton/>为什么就不用设置Autopostpack属性,发现 <asp:LinkButton/> 根本就没有这个属性。(嘿嘿,为什么没有呢,·····有待调查)。回头想想 __doPostBack(){}方法和 Autopostpack属性两个从名字上就是相关联的嘛!
第二章 、  PostBack 的原理
__doPostBack 是一个纯粹并且是非常简单的 javascript 函数,大部分的页面 PostBack 都是由它触发的。注意,这里是 “ 大部分 ” ,因为只有两个 Web Server Control    会自己触发页面的 PostBack, 其它的所有控件都是通过 __doPostBack 函数触发页面的 PostBack ,那先来看一下这个函数的定义吧:
CODE1:
<input type="hidden" value="" />
<input type="hidden" value="" />
function __doPostBack(eventTarget, eventArgument) {
if (!theForm.onsubmit || (theForm.onsubmit() != false)) {
theForm.__EVENTTARGET.value = eventTarget;
theForm.__EVENTARGUMENT.value = eventArgument;
theForm.submit();
}
}
通过上面的代码可以看到, __doPostBack 带有两个参数, eventTarget 是标识将要引发页面 PostBack 的控件 ID , eventArgument 参数提供了在引发页面 PostBack 事件时所带的额外参数。当然这个函数被函数时,这两个参数的值将赋值给页面的两个隐含变量 __EVENTTARGET 和 __EVENTARGUMENT ,然后调用页面的 submit 方法提交页面表单。这就是为什么我们可以通过 Request.Form[“__EVENTTARGET”] 获取得到引发页面 PostBack 的控件 ID 的原因。
了解了 __doPostBack 函数后,我们可以很容易的利用它非常方便地自己触发自定义的 PostBack 事件。那上面也说了,大部分的控件都是调用
析: Request.Form[“__EVENTTARGET”]方法可以查看到触发该事件的控件编号(应该是name才对)(除了button和imagebutton这两种服务器控件,下面会有介绍)。
第三章 Button PostBack 做法
引了页面的 PostBack ,只有两个控件是例外, Button 和 ImageButton ,正是因为它们不是通过调用 __doPostBack 来回发事件,所以通过表单隐含变量 __EVENTTARGET 和 __EVENTARGUMENT 是无法获取得到引发 PostBack 的 Button 或 ImageButton 的 ID 和参数值的,可通过下面的方式实现
1 )在页面中加如 LinkButton ,页面就会在页面中加载 POSTBACK 所需的 JS
<input type="hidden" value="" />
<input type="hidden" value="" />
function __doPostBack(eventTarget, eventArgument) {
if (!theForm.onsubmit || (theForm.onsubmit() != false)) {
theForm.__EVENTTARGET.value = eventTarget;
theForm.__EVENTARGUMENT.value = eventArgument;
theForm.submit();
}
}
2 )利用 GetPostBackEventReference 给客户端生成 __doPostBack()
如:
比如前台页面
< asp:Button  id ="Button1"  runat ="server"  Text ="Button"></ asp:Button >
( 1 )
< a  href ="#"  onclick ="document.getElementById('Button1').click()"> 触发服务器端按钮事件 </ a >
( 2 )
利用 GetPostBackEventReference 给客户端生成 __doPostBack()
前台
< a  href ="#"  onclick ="<%=PostBack()%>"> 触发服务器端按钮事件 </ a >
后台
protected  string  PostBack()
        {
             return  this .Page.GetPostBackEventReference( this .Button1,"haha");
        }
通过 __EVENTARGUMENT="haha" 可以判断是不是点了那个链接的 PostBack 把 Button1 的按钮事件这么写:
if (Request["__EVENTARGUMENT" ]=="haha")
            {
                Response.Write(" 这个是链接的 PostBack");
            }
             else
            {
                Response.Write(" 这个不是链接的 PostBack");
            }

析: 上述主要告诉我们,<asp:button/>和<asp:imagebutton />是不能通过 Request.Form[“__EVENTTARGET”]方法获取到触发控件的编号的。作者告诉了一种验证的方法。但是觉得不太好,自己想了下,之前查过的 UseSubmitBehavior刚好验证这里。
总结上面的,再结合到MSDN其实我们可以得到,button服务器控件是特殊的,它除了通过浏览器来触发意外,还可以通过function __doPostBack(eventTarget, eventArgument){....}这个来实现。那怎么让这个js的方法出来呢,就是呼应了前面的 UseSubmitBehavior=“false”属性。
例2:
前台:
<asp:Button ID="Button4" runat="server" Text="Button" οnclick="Button4_Click" UseSubmitBehavior="false"/>
后台:
protected void Button4_Click(object sender, EventArgs e)
        {
              response.write("ddd");
        }
执行以上代码,查看源文件__doPostBack()方法存在,点击按钮输出ddd,事件方法也执行了。到此两种触发机制了解了。
Bug :
问题:『使用 __doPostBack 会导致』
回发或回调参数无效。在配置中使用 <pages enableEventValidation="true"/> 或在页面中使用 <%@ Page EnableEventValidation="true" %> 启用了事件验证。出于安全目的,此功能验证回发或回调事件的参数是否来源于最初呈现这些事件的服务器控件。如果数据有效并且是预期的,则使用 ClientScriptManager.RegisterForEventValidation 方法来注册回发或回调数据以进行验证。
问题分析及解决方案:『来源网络』
这个要具体分析。本来这个措施是 asp.net2.0 用来防止客户端 “ 欺诈 ” 服务器端的。例如本来输出到客户端的一个事件被触发时需要回发的命令是 “__doPostback('ctl01$abc','user_1')” 的,如果采取采取手段把回发参数由 user_1 改为 user_5 了,服务器端会重新核对输出的是不是 user_5 ,发现和这个页面上一个输出的脚本不一致,就会产生这个异常。
但是,很多程序员写的程序按照过去的习惯(或者按照更加高级灵活的设计例如一些 Ajax 组件)没有考虑这个问题或者是忽略这个欺诈的可能性,写的程序可能会修改参数或者修改目标控件。
因此这样具体问题具体分析。不太可能跟浏览器距离服务器的远近有关,应该还是编程逻辑问题。你应该对出异常的画面以及所使用的数据进行分析。有时候,经常也需要将这个参数设置为 false ,放弃安全管理
以上的内容是对button控件的UseSubmitBehavior属性的了解,其实在之前,还有一个问题也是纠缠不清。就是js中return false;的问题。我们结合到UseSubmitBehavior 看看。会有什么效果。
例3:
<asp:Button ID="Button4" runat="server" Text="Button" οnclick="Button4_Click" UseSubmitBehavior="false" OnClientClick="alert('hello');"/>
后台:
protected void Button4_Click(object sender, EventArgs e)
        {
              response.write("ddd");
        }
与例2相比多了OnClientClick="alert('hello');;执行后查询源代码

<input type="button" name="Button4" value="Button" οnclick="alert('hello');__doPostBack('Button4','')" id="Button4" />

发现__doPostback()js方法在,前面多了句alert('hello');
首先先弹出了hello,(客户端事件),然后输出了ddd(服务器段事件,只不过由客户端触发的).再看下面的例子,
例4:
<asp:Button ID="Button4" runat="server" Text="Button" οnclick="Button4_Click" UseSubmitBehavior="false" OnClientClick="alert('hello');return false;"/>
后台:
protected void Button4_Click(object sender, EventArgs e)
        {
              response.write("ddd");
        }
      与例3相比alert('hello');return false;多了句return false;这也是我们讨论的关键。
执行,查看源代码

<input type="button" name="Button4" value="Button" οnclick="alert('hello');return false;__doPostBack('Button4','')" id="Button4" />
__doPostBack('Button4','')
依然在,前面多了句return false;在看看功能上发现弹出了hello,但是没有输出ddd.这就是我们想要的。

Button的UseSubmitBehavior屬性相关推荐

  1. HTML_DOM的node屬性

    DOM沒搞通,今天繼續學DOM.學三個node屬性,nodename,nodevalue,nodetype屬性.name返回節點名稱,value返回節點值,type返回節點類型.貌似簡單,這裏我用自己 ...

  2. 控制項學習四(屬性與事件)

    四.初探控制項屬性(Property)的建立 1.當我們建立一個控制項(或類別),可以建立控制項的屬性(property),屬性多半是一個值,或是指到另外一個物件,或是一個集合. 請注意,在該屬性宣告 ...

  3. c# 一些控件常用屬性

    Form:     ControlBox:移除窗體按鈕(最大化.最小化.關閉組),並從左側移除「系統菜單」     Opacity:控制窗體透明度     ActiveControl:指出窗體上當前哪 ...

  4. bipush java,Java Opcodes.BIPUSH屬性代碼示例

    本文整理匯總了Java中org.objectweb.asm.Opcodes.BIPUSH屬性的典型用法代碼示例.如果您正苦於以下問題:Java Opcodes.BIPUSH屬性的具體用法?Java O ...

  5. Ios: 如何保護iOS束文件屬性列表,圖像,SQLite,媒體文件

    Ios: 如何保護iOS束文件屬性列表,圖像,SQLite,媒體文件 我創建了Hello World示例項目,然後添加data.plist文件到資源文件夾.現在人們可以很容易得到束文件解壓縮.國際音標 ...

  6. 善用 CSS 中的 table-layout 屬性加快 Table 的顯示速度

    在很久以前我們都是用 Table 在排版的,我相信現在還是有不少人還是在用 Table 進行排版而非現在較為流行的 CSS 排版,使用 Table 排版最大的好處就是版面在各瀏覽器中顯示比較不會亂掉. ...

  7. Mcad学习笔记之通过反射调用類的方法,屬性,字段,索引器(2種方法)

    相关文章导航 Sql Server2005 Transact-SQL 新兵器学习总结之-总结 Flex,Fms3相关文章索引 FlexAir开源版-全球免费多人视频聊天室,免费网络远程多人视频会议系统 ...

  8. java中的terminated_Java State.TERMINATED屬性代碼示例

    本文整理匯總了Java中java.lang.Thread.State.TERMINATED屬性的典型用法代碼示例.如果您正苦於以下問題:Java State.TERMINATED屬性的具體用法?Jav ...

  9. java pixelformat_Java PixelFormat.OPAQUE屬性代碼示例

    本文整理匯總了Java中android.graphics.PixelFormat.OPAQUE屬性的典型用法代碼示例.如果您正苦於以下問題:Java PixelFormat.OPAQUE屬性的具體用法 ...

  10. python keyboard backspace_Python curses.KEY_BACKSPACE屬性代碼示例

    本文整理匯總了Python中curses.KEY_BACKSPACE屬性的典型用法代碼示例.如果您正苦於以下問題:Python curses.KEY_BACKSPACE屬性的具體用法?Python c ...

最新文章

  1. 产品层级提升,产品实战系列之教你如何对货运APP进行运单界面优化
  2. POST 变为AJAX提交代码。
  3. MFC复选框CheckBox使用
  4. java 右移两位_java左移右移运算符
  5. opera在我的博客发不了帖子,郁闷
  6. Nacos服务注册接口
  7. eBPF技术应用云原生网络实践系列之基于socket的service | 龙蜥技术
  8. Shell脚本/bin/bash^M: bad interpreter错误解决方法
  9. python显示乱码_python2显示乱码问题实战详解
  10. 核心动画——CAAnimation
  11. DelphiXE开发坑
  12. plt是什么格式文件
  13. 胡波外挂做模具3D装配的方法
  14. struts的增删查改
  15. HE4484E芯片资料
  16. PTA-IP地址转换
  17. [从头读历史] 第267节 诗经 邶风
  18. 【OpenBMC 系列】2.OpenBMC镜像编译流程
  19. 基于Open CV的植物图像分类识别项目
  20. 5、6月程序员“薪资被应届生倒挂“现象明显,跳槽还是等待?

热门文章

  1. adb脚本选择语句_Python开发:王者荣耀自动刷金币脚本
  2. 羊车门问题看到吐血后的理解----一千七百字小作文+Python代码验证,帮你从质到量上理解,这再看不明白算我输!
  3. 360无痕浏览器_功能强大好用的浏览器
  4. C++“”字符与字符“”相加是连接
  5. Python 核心编程 (全)
  6. 《Ansible权威指南 》一1.6 Ansible应用场景
  7. 2月中国万网域名总量净增2.1万个 份额突破24%
  8. 大型网站架构的演化[转]
  9. Linux文件系统的创建与管理(上)
  10. 加载项目失败的解决办法