由ASP.NET所谓前台调用后台、后台调用前台想到HTTP——实践篇(二)
在由ASP.NET所谓前台调用后台、后台调用前台想到HTTP——理论篇中描述了一下ASP.NET新手的三个问题及相关的HTTP协议内容,在由ASP.NET所谓前台调用后台、后台调用前台想到HTTP——实践篇(一)中已经解答了关于服务器怎么改变页面元素或“调用”JavaScript的问题,为了避免上篇啰啰嗦嗦一大推,这次直奔主题——ASP.NET前台如何调用后台方法
问题细化
所谓前台调用后台这种问题的提出无非是因为很多新手受了ASP.NET服务器端控件的“误导”,你不是说前台没有把JavaScript语句、页面元素什么的传到后台,所以前台调用后台,或者后台监视页面变化是扯淡嘛!那ASP.NET中为什么服务器端控件一点击就能调用后台制定方法呢?按照你前面的理论即使用了服务器端控件,那么到了浏览器页面也变成最普通不过的HTML文本了,怎么还能够调用服务器端方法呢?!
确实,浏览器呈现的是HTML文本,我们先不用拘泥于ASP.NET怎么做到的前台“调用”后台,我们可以自己实现一下,不过在此之前我们得明白一些过于Post的问题
简单聊聊Post
我们在看很多网页的时候都会看到这样的内容
<form id="form1" method="post" action="Default.aspx">
<input type="text" name="wd" />
<input type="submit" name="test" value="test" />
</form>
这段HTML文本同学们都能看懂,页面上有一个form,form内有两个input,点击submit的时候,页面form内容会以post方式被提交到Default.aspx这个页面。
HTTP协议中的Post方法会将所处的form提交到form指定页面,如果form指定的方法为post(还可能是get等),那么form内有些数据会一并被提交。
既然不是整个form被提交,那么哪些数据会被提交呢?
1.input的name和value
2.input type=”submit” 只有被点击的时候才将其本身name与value提交
3.对于多个name相同的input type=”checkbox”,只有选中项的值会被提交(多个value以,连接)
4.对于多个name相同的input type=”radio”,只有选中项的value会被提交(同名的radio只能单选)
5.select的name和被选中的option的vaue(不是option的text)
6.textarea 的name与value
这样被提交的数据可以在提交的页面的后台中以Request.Form[“name”]的方式查到其值
input type=”submit” “调用”后台方法
知道了form和post的最基本知识,我们就可以模拟ASP.NET的button调用后台方法了,首先准备这样一个页面
<form id="form1" method="post" action="Default.aspx">
<input type="submit" name="test" value="test"/>
</form>
在后台添加想调用的方法
protected void test_Click(object sender, EventArgs e)
{
Response.Write("Button Click!");
}
很简单的内容,后台方法的两个参数完全可以不用(和事件有关),填上只是为了增加其逼真性,代码中有几个需要注意的地方
1.form的方法需要设为post,这个很明显。。。
2.form的action设为页面本身,也就是让页面提交给自己
3.input要有name
准备工作做好后就可以模拟JavaScript调用后台方法了,在页面的 OnLoad方法中写入代码
protected override void OnLoad(EventArgs e)
{
if (Request.Form["test"]!=null)
{
test_Click(this, new EventArgs());
}
base.OnLoad(e); }
聪明的同学肯定可以看出端倪了,在点击test的时候会提交表单,页面的OnLoad会自动执行,这时候会判断提交过来的表单中是否包含test(submit被点击后会被提交,不点击不提交,所以可以根据name可以判断出哪个按钮被按了),就这么简单,点击submit的时候,预定义的方法就被调用了,要是页面上有很多submit,这时候就需要if-else或者switch或者for这样的东西来判断到底是哪个submit被点击了,然后调用对应方法。
asp: button
对比一下,ASP.NET正好用,首先人家的from是这样的
<form id="form1" runat="server">
</form>
人家自动就给render成这样了

我们也不需要自己在OnLoad中添加if-else,只要在control的属性中添加OnClick=”XXX”就可以了,我没看过.NET源码,但是我大胆的揣测,虽然不用再OnLoad里if-else了,但应该也是类似的方式处理。
不自动提交表单的怎么办
细心的同学可能要问了,你这么做是可以,但是依赖于表单提交,我们知道并不是所有的页面元素都会点一下就提交表单,像控件asp: DropDownList的OnSelectIndexChanged事件,人家也会调用回台方法,那是怎么做的?
其实非常简单,对于这种需求,仍然可以用表单提交的方式处理,不能自动提交表单,那么我们可以用JavaScript帮助其提交,看个select change “调用”后台函数的例子,首先像这样修改页面,添加一个select
<form id="form1" method="post" action="Default.aspx">
<input type="hidden" id="__param" name="__param" />
<input id="test" name="test" type="submit" value="test" />
<select id="select1" name="select1">
<option value="0">0</option>
<option value="1">1</option>
<option value="2">2</option>
</select>
<script type="text/javascript">
document.getElementById('select1').onchange = function () {
document.getElementById('__param').value = this.name;
document.getElementById('form1').submit();
};
</script>
</form>
我们除了select元素还添加了两个内容,一个input type=”hidden”的隐藏元素,一段JavaScript脚本,在JavaScript脚本中对select的onchange绑定了一个方法,当select选择项变化的时候,把其name赋值给隐藏域的value,然后提交表单,由于隐藏域可定会被提交,这样我们可以同样在页面的Onload中判断谁的onchange触发了,然后调用预定义方法,后台代码是这样的
protected override void OnLoad(EventArgs e)
{
string param= Request.Form["__param"];
if (Request.Form["test"]!=null)
{
test_Click(this, new EventArgs());
}
else if (param == "select1")
{
select_Change(this, new EventArgs());
}
base.OnLoad(e);
} protected void test_Click(object sender, EventArgs e)
{
Response.Write("Button Click!");
} protected void select_Change(object sender, EventArgs e)
{
string newValue = Request.Form["select1"];
Response.Write("Seclect value chaned to " + newValue);
}
是不是很简单呢?先判断是不是submit,不是的话判断隐藏域,命中后调用预定义方法。
细心的同学仍然会发现问题,要是一个元素上同时又两个事件怎么办?比如还想对这个select添加cilck事件怎么办,这时候我们可以修改隐藏域的值,不仅记录name,还记上事件类型,类似这样
document.getElementById('select1').onchange = function () {
document.getElementById('__param').value = this.name+"$ onchange";
document.getElementById('form1').submit();
};
document.getElementById('select1').onclick = function () {
document.getElementById('__param').value = this.name+"$ onclick";
document.getElementById('form1').submit();
};
然后在后台判断的时候,我们可以先把参数按$分割为两个字符串,分别判断元素与事件类型,当然我们也可以再添加一个隐藏域,用来记事件类型,大同小异。
还有同学回问,如果我在客户端同样写了select的onclick,不就把你的代码覆盖了吗?这样我就不能post了,对于这个问题我们大可以换种方式绑定事件处理程序,用addEventListener(attachEven)的方式,这样post的代码和自己写的客户端的就可以同时存在了,这样
var select = document.getElementById('select1');
select.onclick = function () {
//
//
//
};
select.addEventListener('clcik', function () {//浏览器兼容性问题要处理,这里就不演示了
document.getElementById('__param').value = this.name + "$ onclick";
document.getElementById('form1').submit();
}, false);
asp: DropDownList
我们可以看看ASP.NET是怎么处理的,在页面上写一个DropDownList,设起AutoPostBack属性为true
<form id="form1" runat="server">
<asp:DropDownList ID="ddlTest" runat="server" AutoPostBack="true">
<asp:ListItem>1</asp:ListItem>
<asp:ListItem>2</asp:ListItem>
<asp:ListItem>3</asp:ListItem>
</asp:DropDownList>
</form>
生成的页面源码是这样的
<html xmlns="http://www.w3.org/1999/xhtml"><head><title> </title><style type="text/css"></style></head>
<body cz-shortcut-listen="true">
<form method="post" action="Default.aspx" id="form1">
<div class="aspNetHidden">
<input type="hidden" name="__EVENTTARGET" id="__EVENTTARGET" value="">
<input type="hidden" name="__EVENTARGUMENT" id="__EVENTARGUMENT" value="">
</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 class="aspNetHidden"> <input type="hidden" name="__EVENTVALIDATION" id="__EVENTVALIDATION" value="/wEdAAWUpg6MZl3t8/pBYubwqWOJqSwDGOXu15HYfC2efhAyHc1gIfz8erLrnvHjMLCTGHuRsEl8lEzJ1SVLtM7lgCLfggLVPRtDk70yN7TcfPOwwBRLeEs10HSMSLT9zq5aYmmjMFX5QrdQAuJDrw+EXFTJ">
</div>
<select name="ddlTest" onchange="javascript:setTimeout('__doPostBack(\'ddlTest\',\'\')', 0)" id="ddlTest">
<option value="1">1</option>
<option value="2">2</option>
<option value="3">3</option> </select>
</form> </body></html>
主要是这个方法,认真读读是不是和我们刚才写的类似呢,人家传了两个隐藏域,一个用来是Target,一个用来及参数
onchange="javascript:setTimeout('__doPostBack(\'ddlTest\',\'\')', 0)"
function __doPostBack(eventTarget, eventArgument) {
if (!theForm.onsubmit || (theForm.onsubmit() != false)) {
theForm.__EVENTTARGET.value = eventTarget;
theForm.__EVENTARGUMENT.value = eventArgument;
theForm.submit();
}
}
未完待续,与君共勉
通过上面例子分析可以看得出来,JavaScript根本没有调用后台函数,只是提交了含有预设参数的表单,服务器根据参数调用后台方法,生成特定HTML,所谓前台调用后台也就这么回事儿。
有一个问题值得思考,为什么这种很想当然的做法PHP程序员可以轻松想到,很多.NET初级程序员却不行,甚至有时候我们都已经在使用了,比如根据Url的QueryString来调用不同方法,或进行不同处理,却没有进一步想到更常用的Button、DropDownList是怎么做到的?还是那句话,.NET程序员当自勉,不要被.NET的易用性惯坏,多学习原理知识,不要过分满足于做出来,而要多追求为什么,这样才能不断进步。
由ASP.NET所谓前台调用后台、后台调用前台想到HTTP——实践篇(二)的更多相关文章
- ASP.NET前台JS与后台CS函数如何互相调用
摘要: 在实际的Web开发中,我们可能会常常遇到后台调用前台JS代码或者前台JS调用后台代码的情况.今天就把比较实用的前后台相互调用的方法总结出来和大家分享. 在实际的Web开发中,我们可能会常常遇到 ...
- ASP.NET所谓前台调用后台、后台调用前台想到HTTP——实践篇
由ASP.NET所谓前台调用后台.后台调用前台想到HTTP——实践篇 在由ASP.NET所谓前台调用后台.后台调用前台想到HTTP——理论篇中描述了一下ASP.NET新手的三个问题及相关的HTTP协议 ...
- js调用.net后台事件、后台调用前台以及js调用服务器控件
1. javaScript函数中执行C#代码中的函数: 方法一:间接触发后台代码 1.首先建立一个服务端控件按钮命名为btn1,双击进入后台将调用或处理的内容写入btn1_click中; 2.在前台写 ...
- Asp.net中前台javascript与后台C#交互
方法一:使用Ajax开发框架,后台方法定义前添加[AjaxPro.AjaxMethod],然后就可以在前台js脚本中调用后台C#函数. 方法二:后台方法声明为public或者protected,然后前 ...
- js调用.net后台事件,和后台调用前台等方法以及js调用服务器控件的方法
http://blog.csdn.net/deepwishly/article/details/6670942 ajaxPro.dll基础教程(前台调用后台方法,后台调用前台方法) 1. javaS ...
- Asp.Net Core 2.0 项目实战(10) 基于cookie登录授权认证并实现前台会员、后台管理员同时登录
1.登录的实现 登录功能实现起来有哪些常用的方式,大家首先想到的肯定是cookie或session或cookie+session,当然还有其他模式,今天主要探讨一下在Asp.net core 2.0下 ...
- ASP.NET前台代码绑定后台变量方法总结
经常会碰到在前台代码中要使用(或绑定)后台代码中变量值的问题.一般有<%= str%>和<%# str %>两种方式,这里简单总结一下.如有错误或异议之处,敬请各位指教. 一方 ...
- 转:ASP.NET前台代码绑定后台变量方法总结
经常会碰到在前台代码中要使用(或绑定)后台代码中变量值的问题.一般有<%= str%>和<%# str %>两种方式,这里简单总结一下.如有错误或异议之处,敬请各位指教. 一方 ...
- [ASP.NET] 前台代码绑定后台变量方法总结 [转]
原文链接:https://www.cnblogs.com/lerit/archive/2010/10/22/1858007.html 经常会碰到在前台代码中要使用(或绑定)后台代码中变量值的问题.一般 ...
随机推荐
- mysql galera cluster 集群的分裂与仲裁机制
集群的分裂 当集群由于网络原因分裂为几个单独的组时(一组可能是单节点,也可能是几个互联的节点),数据出现不一致,此时可能产生脑裂及数据不一致.这种情况 下,只有一组节点能够继续提供服务,这组节点的状态 ...
- php socket解决方案
最近一直在为移动应用提供 php服务端api,以前 实时交互数据需求不严格(定时从手机端发送http请求),现在业务需求变更, 需要实时交互式接口,必须增加socket. 服务端框架使用YII 1.1 ...
- 想在Images.xcassets 只能用 imageNamed 加载里边的素材 其他方法 你就别费老劲了
1.Images.xcassets中的素材 (1)只支持png格式的图片 (2) 图片只支持[UIImage imageNamed]的方式实例化,但是不能从Bundle中加载 (3) 在编译时,Im ...
- 在項目中快速部署SLF4J+LOGBACK
想了解SLF4J,LOGBACK是什么?可以访问:http://www.slf4j.org/ http://logback.qos.ch/ 本文大部分参考了Cody Burleson<Ho ...
- 响应式web设计之CSS3 Media Queries
开始研究响应式web设计,CSS3 Media Queries是入门. Media Queries,其作用就是允许添加表达式用以确定媒体的环境情况,以此来应用不同的样式表.换句话说,其允许我们在不改变 ...
- 在 Mac OS X 上创建的 .NET 命令行程序访问数据库 (使用Entity Framework 7 )
var appInsights=window.appInsights||function(config){ function r(config){t[config]=function(){var i= ...
- 看来System.dll是没法剔除依赖了
今天花了半天时间将System.Xml换成了Mono.Xml 想干掉System.dll发现不行了,System.Net以及System.IO都在这下面,还有protobuf-net也逃不掉这个 算啦 ...
- Java日志系统及框架分析
最近在考虑将容器(Tomcat)内的应用日志统一成slf4j + logback,主要目的有: 快速定位应用日志输出路径,方便日志的采集: 能动态调整日志的级别,方便线上问题定位: 方便在容器层面做扩 ...
- KIWI Syslog配置
日志服务器Kiwi+Syslogd+8.3.7破解版 Window收集服务器日志evtsys_exe_32 默认地,kiwi使用UDP 514端口接收日志数据,安装成功后即可接收日志 使用命令nets ...
- js中,var 修饰变量名和不修饰的区别
js中 允许在定义变量的时候 不加var 修饰符.js会在当前作用域下寻找上下文是否定义了此变量, 如果没有找到则会为这个变量分配内存.当且将其视为window的成员. 也就是全局变量. 如果加了va ...