这里先不说标题上的UseSubmitBehavior属性是什么,先说下面这种情况。

通常,在我们写一个表单页面的时候,最下方会有“提交”和“返回”字样的两个按钮。顾名思义,它们的功能大家都知道,但是一般情况下我们会给表单的内容加上一些验证,这样就出现了一个问题。因为两个按钮是服务器控件(有runat=”Server”属性),所以点击按钮之后会先进行验证(无论这里你用的是前台的jQuery.validate验证或者是ASP.NET自带的后台验证控件验证,都会先进行验证)。对于“提交”按钮,这确实是我们想要的,但是对于“返回”按钮,我们并不想出现这种情况,而是想让它不经过验证,直接跳转回上一页。

对于这种情况,我以前的解决方法就是把

<asp:Button ID="button_back" runat="server" Text="back" OnClick="button_back_Click" />

换成

<input type="button" value="back" onclick="BackToPage();" />

这种形式。BackToPage方法实现页面跳转。

我相信很多朋友都是这么写的吧?反正能解决问题就行。为什么?往下看。

但是今天无意中发现了Button.UseSubmitBehavior这个属性,将其置成false,会使“返回”按钮“躲避”掉表单验证,直接执行点击事件中的方法。这正是我们一直以来想要的不是么?而且按钮还很统一,不用去加额外的js代码。

这里要说明一下,大家不要以为Button.UseSubmitBehavior真的是用来“躲避”验证的,我只是用这个词来表达一下它实现的效果。实际上Button控件有专门的属性是用来屏蔽验证控件的。就是Button.CausesValidation,通过名字我们应该就能了解一二。

那你会不会问,“既然有这个属性,以上问题就不算是问题啦,可以很容易的解决。”

我想说,“是这样的啊!”但前提是,你的项目中只存在后台验证。但是现实中我觉得不会所有人都这样做吧?反正我就蛮习惯用jQuery.validate验证插件的(前台验证)。那我们继续研究Button.UseSubmitBehavior吧。

看个例子:

前台代码:

<asp:Button ID="button_confirm" runat="server" Text="确定" />
<asp:Button ID="button_back" runat="server" Text="返回" onclick="button_back_Click" />

浏览器中查看源码中的形式

前台代码:

<asp:Button ID="button_confirm" runat="server" Text="确定" />
<asp:Button ID="button_back" runat="server" Text="返回" UseSubmitBehavior="false" onclick="button_back_Click" />

浏览器中查看源码中的形式

大家可以看出加上UseSubmitBehavior属性后,解析成的html语句明显不同。看到这我才恍然大悟,当加过UseSubmitBehavior属性之后,type属性变成了button,这样就跟以往的解决方法一样了啊,因为不是submit属性,所以也就不会触发jQuery.validate的验证方法(回答上面橙色加粗部分的为什么)。这就是为什么UseSubmitBehavior 置成false后,不用去进行前台验证的原因。这就是谜底。

但是到这就结束了会不会觉得有些怪呢?我们为了达到我们的目的,使用了一个新的属性,但是这个属性存在的意义并非是为了达到我们的目的。

整理一下思绪,上面的例子我们需要从脑袋里抹去。UseSubmitBehavior属性与是否触发表单验证完全没有任何关系。我们来专注这个属性的本身,毕竟这个是标题。

我很俗的查了一下MSDN

可以看出重点就是图里指出的两个机制:

1、客户端浏览器的提交机制

2、ASP.NET 回发机制

解释一下,有错误的话请大家指正

1、大家可以看下这个链接所讲的第一部分:http://www.th7.cn/Program/net/201309/150415.shtml

就是浏览器会封装一个请求报文,发给服务器,服务器解析这个报文,进行重组,生成一个响应报文,回发给浏览器,浏览器收到后再对其进行解析,就生成了我们看到的网页和一些我们看不到的数据。它们之间的通信都是遵循HTTP协议。

重点强调:HTTP是无状态协议,也就是说每次的浏览器请求,服务器响应都是全新的一次。

2、大家可以看下这个链接所讲的内容:http://blog.sina.com.cn/s/blog_7815564501012qgy.html   对于回发机制,我希望大家能跟着这篇文章里讲的步骤写个小Demo试试,会更有感觉。

回发机制,就是自己请求自己的页面,这个机制比较搞,如果大家理解上一个机制,就会发现ASP.NET回发机制与其是有点背离的,上一个是无状态的,这一个是能将前一次的值保存住,作为下一次页面的初值。

举个实际中的例子,就是当我们在填写一个表单时,有好多好多项,但是当我们填到倒数第二项的时候,不小心点击了刷新,正常按第一个机制来说,表单内控件上的值应该全部被清空,因为此时还没有做数据库读取的操作,所以页面不可能有值,但是有了回发机制,__viewstate把页面保存了下来,这样就保留下了我们先前填写的内容。不过技术群里的一个朋友说“现在这个东西已经很少会用了,基本不会用控件做项目开发。”囧rz啊~我一直都用的不亦乐乎,怎么破?

两个机制解释完了,可能大家还是有点一头雾水,其实此时我也是一样。下面写个Demo让整件事情变得更清晰点吧,借用一下Fiddler 这个工具。

页面展示:

html代码:

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
<title></title>
</head>
<body>
<form id="form1" runat="server" method="post">
<div>
<asp:Label ID="Label1" runat="server" Text="content:" ></asp:Label>
<asp:TextBox ID="textbox_content" runat="server" width="500px"></asp:TextBox>
<asp:Button ID="button_usesubmitbehavior_true" runat="server" Text="browser-server"
onclick="button_usesubmitbehavior_true_Click" />
<asp:Button ID="button_usesubmitbehavior_false" runat="server" Text="postback"
UseSubmitBehavior="false" onclick="button_usesubmitbehavior_false_Click" />
</div>
</form>
</body>
</html>

页面源码:

后台代码:

    public partial class WebForm1 : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
textbox_content.Text += "M";
} protected void button_usesubmitbehavior_true_Click(object sender, EventArgs e)
{
textbox_content.Text += "true";
button_usesubmitbehavior_true.Text += "1";
button_usesubmitbehavior_false.Text += "2";
} protected void button_usesubmitbehavior_false_Click(object sender, EventArgs e)
{
textbox_content.Text += "false";
button_usesubmitbehavior_true.Text += "3";
button_usesubmitbehavior_false.Text += "4";
} }

操作步骤:

1、初始页面

2、先连续点击browser-server按钮三次

3、再连接点击postback按钮三次。

通过这个demo示例,如果单从页面显示上看,我们完全看不到两者有什么不同,因为后台想实现的功能两者都实现了(就是点击一下按钮,会经过一次page_load方法,然后再经过各自的click事件,给文本框和两个按钮追加内容)。但是利用fiddler工具,我们会发现提交的表单内容的确是有些不一样,UseSubmitBehavior属性设为false的按钮提交时,本身没有作为表单的参数传递给服务器端。

这就是唯一的不同吗?感觉还是不够,不过我真的是写不动了。

有兴趣的朋友可以再想想别的对比示例,比如试试在page上加EnableViewState="false" 这个属性,看看点击完按钮会是什么效果的。这个我也有试,不过感觉还是不能证明什么。

啊,最后终于想到了一个,你可以给browser-server按钮加上EnableViewState="false"属性,嗯嗯,这个比较靠谱,给browser-server 按钮加上EnableViewState="false"属性 ,目的就是让其没有回发机制,因为Asp.Net控件默认都会有回发机制,这样browser-server 按钮就只有单纯的客户端浏览器的提交机制,而postback按钮就只有回发机制。页面效果异常明显啊~

卖个关子,想看效果的朋友自己去试一下吧,大家可以留言做交流,原谅我,这篇文章写的属实有点累。

总结一下,UseSubmitBehavior 这个属性绝对可以说是非主流的,花时间去研究会不会很蛋疼?

其实最初根本没有想过这篇文章会写这么长,但是随着研究这个属性深入,我去学习了ASP.NET的底层交互原理,ASP.NET页面生命周期,回发机制以及了解了验证相关的问题。说实话,收获蛮多的,也很开心。当然,还要继续实践,学习。

附件的链接:http://files.cnblogs.com/zhouhongyu1989/UseSubmitBehavior%E7%A4%BA%E4%BE%8BDemo.rar

ASP.NET Button控件的UseSubmitBehavior属性引发的血案的更多相关文章

  1. asp.net button控件 使用JS的 disabled

     今天想用JS禁用asp.net的button控件,查了好久,都是一行代码....      document.getElementById("Button1").disabled ...

  2. 关于Button控件的CommandName属性用法的一个实例

    注:本文分享于悠闲的博客,地址:http://www.cnblogs.com/9999/archive/2009/11/24/1609234.html 1.前台的代码 <%@ Page Lang ...

  3. asp.net GridView控件的列属性

    BoundField 默认的数据绑定类型,通常用于显示普通文本 CheckBoxField 显示布尔类型的数据.绑定数据为TRUE时,复选框数据绑定列为选中状态:绑定数据为FALSE时,则显示未选中状 ...

  4. ASP.NET服务器端控件(class0617)

    ASP.Net服务端基本控件介绍 ASP.Net服务端控件是ASP.Net对HTML的封装,在C#代码中就可以用txt1.Text=‘abc’这种方式来修改input的值,ASP.Net会将服务端控件 ...

  5. C# Windows - Button 控件

    .Net Framework提供了一个派生于Control的类System.Windows.Forms.ButtonBase,它实现了Button控件所需的基本功能. System.Windows.F ...

  6. asp.net动态生成按钮Button控件

    1.动态生成button控件及响应服务端和客户端事件 void BindButtons(){ foreach (var item in items) { Button Btn = new Button ...

  7. ASP.Net UpdatePanel控件(转)

      Asp.net UpdatePanel 允许用户构建一个丰富的,以客户端为中心的应用程序,引用UpdatePanel控件,能够实现页面的部分刷新,一个包含scriptManage和 UpdateP ...

  8. ASP.Net 验证控件 RequiredFieldValidator

    使用 ASP.NET 验证控件可在网页上检查用户输入.有用于各种不同类型验证的控件,例如范围检查或模式匹配验证控件.每个验证控件都引用网页上其他位置的输入控件(服务器控件).当处理用户输入时(例如,当 ...

  9. ASP.NET数据绑定控件

    数据绑定控件简介 数据绑定分为:数据源 和 数据绑定控件 两部分,数据绑定控件通过数据源来获得数据,通过数据源来隔离数据提供者和数据使用者,数据源有:SqlDataSource,AccessDataS ...

随机推荐

  1. 深入了解java虚拟机(JVM) 第十三章 虚拟机字节码执行引擎

    一.概述 执行引擎是java虚拟机最核心的组成部件之一.虚拟机的执行引擎由自己实现,所以可以自行定制指令集与执行引擎的结构体系,并且能够执行那些不被硬件直接支持的指令集格式.所有的Java虚拟机的执行 ...

  2. CentOS6.5下openssh服务

    00×0 介绍 OpenSSH是使用SSH通过计算机网络加密通讯的实现.它是取代由SSH Communications Security所提供的商用版本的开放源代码方案.目前OpenSSH是OpenB ...

  3. unittest测试框架和测试报告的输出实例(一)

    我们整个自动化才是报告的环节基本上分为三个部分: 1.测试用例的准备 2.测试用例的执行 3.测试报告的输出 1.测试用例的准备: 那我们就以搜孤网页做一个简单的用例: from selenium i ...

  4. python 对mongodb进行压力测试

    最近对mongoDB数据库进行性能分析,需要对数据库进行加压. 加压时,最初采用threading模块写了个多线程程序,测试的效果不理想. 单机读数据库每秒请求数只能达到1000次/s.而开发的jav ...

  5. java_I/O字节流

    I/O流(Stream) INPUT:输入流,从文件里读OUPUT:输出流,写内容到文件 IO流分为:字符流和字节流 字符流:处理纯文本文件. 字节流:处理可以所有文件. 测试字节输出流OuPut(写 ...

  6. java中的jdk配置详解:

    1.配值系统变量"JAVA_HOME" 变量名JAVA_HOME: 指向:JDK(java开发工具包)的安装路径 目的:使用JDK安装目录时,可以直接通过”%JAVA_HOME%“ ...

  7. leetcode-383-Ransom Note(以空间换时间)

    题目描述: Given an arbitrary ransom note string and another string containing letters from all the magaz ...

  8. asp.net图片上传代码

    前端: <form action="/ImageUpload.ashx" method="post" enctype="multipart/fo ...

  9. ASP.NET MVC 下拉列表实现

    https://blog.csdn.net/Ryan_laojiang/article/details/75349555?locationNum=10&fps=1 前言 我们今天开始好好讲讲关 ...

  10. java 调用 C# 类库 实战视频, 非常简单, 通过 云寻觅 javacallcsharp 生成器 一步即可!

    java 调用 C# 类库 实战视频, 非常简单, 通过 云寻觅 javacallcsharp 生成器 一步即可! 通过 云寻觅 javacallcsharp 生成器 自动生成java jni类库,  ...