Fork me on GitHub

  我们在Web开发中经常会接触到Ajax技术,同时Ajax技术也有很多种实现方式,那么,我们今天从第一种方式说起:ASP.NET原生控件实现Ajax。

  ASP.NET原生控件用于Ajax技术的主要是UpdatePanel和ScriptManager,前者顾名思义,是一个可以用于盛放内容的容器,用于实现页面的局部更新,在使用的时候直接将需要更新的内容放入即可。后者用于调用一些服务和脚本:例如我们本次Demo中使用JavaScript调用WebService服务。

  本次Demo主要包含一个页面,一个Web服务(WebService)和一个数据通用操作类(userHelper)来实现页面无刷新检测注册时昵称是否存在的功能。

  首先,我们需要建立一个Web页面(此处选用WebForm),大致代码如下:

  

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="WebForm2.aspx.cs" Inherits="WebApplication1.WebForm2" %>

<!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 runat="server">
    <title>注册</title>
    <script type="text/javascript">
        function check() {
            var Name = document.getElementById("txtName").value;
            if (Name == "") {
                alert("昵称不能为空!");
            }
            else {
                WebApplication1.WebService1.CheckName(Name, callbackCheck);
                function callbackCheck(result) {
                    if (result == false) {
                        document.getElementById("checkFont").innerHTML = "<span>该昵称已存在!请重新输入!</span>";
                    }
                    else {
                        document.getElementById("checkFont").innerHTML = "<span>恭喜!该昵称可以使用!</span>";
                    }
                }
            }

        }
    </script>
</head>
<body>
    <form id="form1" runat="server">
    <asp:ScriptManager runat="server" ID="scriptManager">
        <Services>
            <asp:ServiceReference Path="~/WebService1.asmx" />
        </Services>
    </asp:ScriptManager>
    <asp:UpdatePanel ID="Upnl" runat="server" UpdateMode="Always">
        <ContentTemplate>
            <div>
                昵称:<asp:TextBox ID="txtName" runat="server"></asp:TextBox><div id="checkFont">
                </div>
            </div>
        </ContentTemplate>
        <Triggers>
            <asp:AsyncPostBackTrigger ControlID="btnCheck" />
            <asp:PostBackTrigger ControlID="btnReFresh" />
        </Triggers>
    </asp:UpdatePanel>
    <asp:Button ID="btnCheck" runat="server" OnClientClick="check();" Text="检测昵称" />
    <asp:Button ID="btnReFresh" runat="server" Text="刷新页面" />
    </form>
</body>
</html>

  我在这个页面中放入了一个ScriptManager,同时引用了一个WebService,然后将需要更新的部分放入了UpdatePanel。此外还有两个Button,第一个用于检测昵称,第二个用于刷新页面。大家看过代码之后会发现我在UpdatePanel设置时加入了一个Triggers,那么这个是干什么的呢?获取或设置触发对 UpdatePanel 控件进行更新的回发控件事件。可以看到,我将检测昵称的Button设置为AsyncPostBackTrigger模式,此为点击按钮时不刷新整个页面,只刷新UpdatePanel中的内容,而第二个按钮设为PostBackTrigger模式,则会刷新整个页面。

  在第一个检测昵称的按钮点击时会调用一个js方法:check(),在check()函数中,我们首先去获取TextBox中用户输入的名称,然后将名称传入js调用的WebService方法中进行操作,得到结果后再根据结果去设置div要显示的内容,如果存在则返回值为false。这里有一个js回调函数的概念:回调函数用于在异步操作中扩展第一个函数的功能,我们此处是为了在返回结果之后设置div的innerHTML的值。如果不使用回调函数而直接使用同步的话,可能会不执行函数的内容。

      接下来是WebService,在建立WebService时,需要注意将[System.Web.Script.Services.ScriptService]这行代码的注释取消掉,VS2010中建立后会提示使用ASP.NET

Ajax从脚本中调用此Web服务时就需要这样,否则会调用不到该方法。WebService中代码如下:

  

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Services;

namespace WebApplication1
{
    /// <summary>
    /// WebService1 的摘要说明
    /// </summary>
    [WebService(Namespace = "http://tempuri.org/")]
    [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
    [System.ComponentModel.ToolboxItem(false)]
    // 若要允许使用 ASP.NET AJAX 从脚本中调用此 Web 服务,请取消对下行的注释。
    [System.Web.Script.Services.ScriptService]
    public class WebService1 : System.Web.Services.WebService
    {

        [WebMethod]
        public string HelloWorld()
        {
            return "Hello World";
        }
        [WebMethod]
        public bool CheckName(string Name) 
        {
            return userHelper.CheckName(Name);
        }
    }
}

  我们此处调用的CheckName方法,源自于数据通用类中的CheckName方法,userHelper中实现方法代码如下:

  

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Data;
using System.Data.SqlClient;

namespace WebApplication1
{
    public class userHelper
    {


        public static bool CheckName(string Name)
        {
            string Con = @"Data Source=205ZHANGXK\SQLEXPRESS;Initial Catalog=MyQQ;Connect Timeout=30;Trusted_Connection=true";
            using (SqlConnection sqlCon = new SqlConnection(Con))
            {
                string Sql = "select count(*) from users where NickName='" + Name + "'";
                sqlCon.Open();
                SqlCommand cmd = sqlCon.CreateCommand();
                cmd.CommandText = Sql;
                int res = int.Parse(cmd.ExecuteScalar().ToString());
                if (res == 1)
                {
                    return false;
                }
                else
                {
                    return true;
                }
               
            }
        }
    }
}

  我在操作时连接了本地的数据库MyQQ,该方法实现即为查找名称,然后返回受影响的行数并返回结果。下面我们开始调试代码:

  

  我们可以看到数据库中已经存在如下11条昵称,在运行时我们输入“豆豆”并点击检测昵称按钮,会发现代码进入到WebSerivce中:

  

  此时说明js脚本调用WebService方法已经成功,接下来会执行通用操作类中的方法:

  

  和我们意料之中相同,“豆豆”这个昵称已经在数据库存在,接下来我们看看最终的结果:

  

  在页面上已经出现了我们预期的结果,我们再来试试输入另外一个不存在的昵称:

  

  也已经达到了我们预期的效果。由于局部刷新的图不太容易截到,这里就无法为大家呈现了。我们大家都知道服务器端Button在点击时都会出现响应回发,因此点击另外一个刷新页面的按钮,页面会有明显的闪烁(整个页面全部刷新),这也体现了Ajax技术带来的优势,提升了用户的体验度。

  我们通过一个简单的Demo重现了ASP.NET 原生控件实现Ajax的功能,这也体现了ASP.NET本身封装的强大之处,同时使用起来也很方便。但是由于ScriptManager和UpdatePanel控件本身为服务器端控件,在一定程度上会生成一些垃圾代码,而且在请求时会生成整个页面的控件树,因此在批量使用的情况下会导致资源的占用率很高,这也是它本身的缺点。下次我会为大家带来Ajax的另一种实现方式:JQuery自带的ajax()方法结合ashx一般处理程序实现,这是一种轻量级的实现方式。

posted on 2014-06-28 16:53  Wackysoft  阅读(1546)  评论(5编辑  收藏  举报