掌握AJAX第5/7页

 更新时间:2006年09月17日 00:00:00   作者:  
学习
  • 您可以参阅本文在 developerWorks 全球站点上的 英文原文
  • 学习介绍 Ajax 的本 developerWorks 系列的前几篇文章:
  • 掌握 Ajax,第 1 部分:Ajax 简介” 第 1 部分演示 Ajax 组件技术如何一起工作,并揭露 Ajax 的 核心概念,其中包括 XMLHttpRequest 对象(2006 年 1 月)。
  • 掌握 Ajax,第 2 部分: 使用 JavaScript 和 Ajax 发出异步请求” 第 2 部分展示如何以跨浏览器方式创建 XMLHttpRequest 实例、构造和发送请求,以及响应服务器(2006 年 2 月)。
  • 掌握 Ajax,第 3 部分: Ajax 中的高级请求和响应” 第 3 部分演示了 Ajax 中标准 Web 表单如何执行,并展示了如何掌握对 HTTP 状态代码、就绪态以及 XMLHttpRequest 对象的理解(2006 年 3 月)。
  • 万维网联盟上的 DOM 主页:所有 DOM 相关内容的入口。
  • DOM Level 3 Core Specification:定义核心 Document Object Model,涵盖了可用类型和属性以及在各种语言中 DOM 的用法。
  • ECMAScript language bindings for DOM:如果您是 JavaScript 程序员并想在代码中使用 DOM,Level 3 Document Object Model Core 定义的本附录将投您所好。
  • 面向 Java 开发人员的 Ajax: 构建动态的 Java 应用程序”:本文从 Java 角度来观察服务器端的 Ajax,介绍了一种用于创建动态 Web 应用程序体验的突破性方法(developerWorks,2005 年 9 月)。
  • 面向 Java 开发人员的 Ajax: Ajax 的 Java 对象序列化”:完整介绍了 Java 对象序列化的五种方法,并讨论了如何通过网络发送对象以及如何与 Ajax 交互(developerWorks,2005 年 10 月)。
  • 使用 AJAX 调用 SOAP Web 服务,第 1 部分: 构建 Web 服务客户机”:这篇相当高级的文章深入探讨了将 Ajax 与现有的基于 SOAP 的 Web 服务相集成;展示了如何使用 Ajax 设计模式实现基于 Web 浏览器的 SOAP Web 服务客户机(developerWorks,2005 年 10 月)。
  • Ajax: A New Approach to Web Applications”:本文创造了 Ajax 这个名字,它是所有 Ajax 开发人员的必读之作。
  • HTTP 状态代码:从 W3C 查看状态代码的完整清单。
  • Head Rush Ajax,作者 Elisabeth Freeman、Eric Freeman 和 Brett McLaughlin(2006 年 2 月,O'Reilly Media, Inc.):以深入浅出的方式将本文的概念装进您的大脑。
  • Java and XML,第二版,作者 Brett McLaughlin(2001 年 8 月,O'Reilly Media, Inc.):阅读作者对 XHTML 和 XML 转换的讨论。
  • JavaScript: The Definitive Guide,作者 David Flanagan(2001 年 11 月,O'Reilly Media, Inc.):提供使用 JavaScript 和动态 Web 页面的全面说明。未来版本将增加关于 Ajax 的两章。
  • Head First HTML with CSS & XHTML,作者 Elizabeth 和 Eric Freeman(2006 年 12 月,O'Reilly Media, Inc.):细读完整的代码,学习 XHTML、CSS 以及如何结合使用。
  • developerWorks Web 体系结构专区:拓展您的 Web 构建能力。
  • developerWorks technical events and Webcasts:为技术开发人员提供这些软件的最新简讯。
  • ax 编程来说证明是一种短视而错误的方法。如果脚本需要认证,而请求却没有提供有效的证书,那么服务器就会返回诸如 403 或 401 之类的错误代码。然而,由于服务器对请求进行了应答,因此就绪状态就被设置为 4(即使应答并不是请求所期望的也是如此)。最终,用户没有获得有效数据,当 JavaScript 试图使用不存在的服务器数据时就可能会出现严重的错误。

    它花费了最小的努力来确保服务器不但完成了一个请求,而且还返回了一个 “一切良好” 的状态代码。这个代码是 "200",它是通过 XMLHttpRequest 对象的 status 属性来报告的。为了确保服务器不但完成了一个请求,而且还报告了一个 OK 状态,请在您的回调函数中添加另外一个检查功能,如 清单 7 所示。

    清单 7. 检查有效状态代码

       function updatePage() {
         if (request.readyState == 4) {
           if (request.status == 200) {
             var response = request.responseText.split("|");
             document.getElementById("order").value = response[0];
             document.getElementById("address").innerHTML =
               response[1].replace(/\n/g, "<br />");
           } else
             alert("status is " + request.status);
         }
       }

    通过添加这几行代码,您就可以确认是否存在问题,用户会看到一个有用的错误消息,而不仅仅是看到一个由断章取义的数据所构成的页面,而没有任何解释。

    重定向和重新路由

    在深入介绍有关错误的内容之前,我们有必要来讨论一下有关一个在使用 Ajax 时 并不需要 关心的问题 —— 重定向。在 HTTP 状态代码中,这是 300 系列的状态代码,包括:

    ·301:永久移动
    ·302:找到(请求被重新定向到另外一个 URL/URI 上)
    ·305:使用代理(请求必须使用一个代理来访问所请求的资源)

    Ajax 程序员可能并不太关心有关重定向的问题,这是由于两方面的原因:

    ·首先,Ajax 应用程序通常都是为一个特定的服务器端脚本、servlet 或应用程序而编写的。对于那些您看不到就消失了的组件来说,Ajax 程序员就不太清楚了。因此有时您会知道资源已经移动了(因为您移动了它,或者通过某种手段移动了它),接下来要修改请求中的 URL,并且不会再碰到这种结果了。
    更为重要的一个原因是:Ajax 应用程序和请求都是封装在沙盒中的。这就意味着提供生成 Ajax 请求的 Web 页面的域必须是对这些请求进行响应的域。因此 ebay.com 所提供的 Web 页面就不能对一个在 amazon.com 上运行的脚本生成一个 Ajax 风格的请求;在 ibm.com 上的 Ajax 应用程序也无法对在 netbeans.org 上运行的 servlets 发出请求。
    ·结果是您的请求无法重定向到其他服务器上,而不会产生安全性错误。在这些情况中,您根本就不会得到状态代码。通常在调试控制台中都会产生一个 JavaScript 错误。因此,在对状态代码进行充分的考虑之后,您就可以完全忽略重定向代码的问题了。

    结果是您的请求无法重定向到其他服务器上,而不会产生安全性错误。在这些情况中,您根本就不会得到状态代码。通常在调试控制台中都会产生一个 JavaScript 错误。因此,在对状态代码进行充分的考虑之后,您就可以完全忽略重定向代码的问题了。

    错误

    一旦接收到状态代码 200 并且意识到可以很大程度上忽略 300 系列的状态代码之后,所需要担心的唯一一组代码就是 400 系列的代码了,这说明了不同类型的错误。回头再来看一下 清单 7,并注意在对错误进行处理时,只将少数常见的错误消息输出给用户了。尽管这是朝正确方向前进的一步,但是要告诉从事应用程序开发的用户和程序员究竟发生了什么问题,这些消息仍然是没有太大用处的。

    首先,我们要添加对找不到的页的支持。实际上这在大部分产品系统中都不应该出现,但是在测试脚本位置发生变化或程序员输入了错误的 URL 时,这种情况并不罕见。如果您可以自然地报告 404 错误,就可以为那些困扰不堪的用户和程序员提供更多帮助。例如,如果服务器上的一个脚本被删除了,我们就可以使用 清单 7 中的代码,这样用户就会看到一个如 图 5 所示的非描述性错误。

    边界情况和困难情况

    看到现在,一些新手程序员就可能会这究竟是要讨论什么内容。有一点事实大家需要知道:只有不到 5% 的 Ajax 请求需要使用诸如 2、3 之类的就绪状态和诸如 403 之类的状态代码(实际上,这个比率可能更接近于 1% 甚至更少)。这些情况非常重要,称为 边界情况(edge case) —— 它们只会在一些非常特殊的情况下发生,其中遇到的都是最奇特的问题。虽然这些情况并不普遍,但是这些边界情况却占据了大部分用户所碰到的问题的 80%!

    对于典型的用户来说,应用程序 100 次都是正常工作的这个事实通常都会被忘记,然而应用程序只要一次出错就会被他们清楚地记住。如果您可以很好地处理边界情况(或困难情况),就可以为再次访问站点的用户提供满意的回报。

    图 5. 常见错误处理

    用户无法判断问题究竟是认证问题、没找到脚本(此处就是这种情况)、用户错误还是代码中有些地方产生了问题。添加一些简单的代码可以让这个错误更加具体。请参照 清单 8,它负责处理没找到的脚本或认证发生错误的情况,在出现这些错误时都会给出具体的消息。

    清单 8. 检查有效状态代码

       function updatePage() {
         if (request.readyState == 4) {
           if (request.status == 200) {
             var response = request.responseText.split("|");
             document.getElementById("order").value = response[0];
             document.getElementById("address").innerHTML =
               response[1].replace(/\n/g, "<br />");
           } else if (request.status == 404) {
             alert ("Requested URL is not found.");
           } else if (request.status == 403) {
             alert("Access denied.");
           } else
             alert("status is " + request.status);
         }
       }

    虽然这依然相当简单,但是它的确多提供了一些有用的信息。图 6 给出了与 图 5 相同的错误,但是这一次错误处理代码向用户或程序员更好地说明了究竟发生了什么。

    图 6. 特殊错误处理

    在我们自己的应用程序中,可以考虑在发生认证失败的情况时清除用户名和密码,并向屏幕上添加一条错误消息。我们可以使用类似的方法来更好地处理找不到脚本或其他 400 类型的错误(例如 405 表示不允许使用诸如发送 HEAD 请求之类不可接受的请求方法,而 407 则表示需要进行代理认证)。然而不管采用哪种选择,都需要从对服务器上返回的状态代码开始入手进行处理。

    其他请求类型

    如果您真希望控制 XMLHttpRequest 对象,可以考虑最后实现这种功能 —— 将 HEAD 请求添加到指令中。在前两篇文章中,我们已经介绍了如何生成 GET 请求;在马上就要发表的一篇文章中,您会学习有关使用 POST 请求将数据发送到服务器上的知识。不过本着增强错误处理和信息搜集的精神,您应该学习如何生成 HEAD 请求。

    生成请求

    实际上生成 HEAD 请求非常简单;您可以使用 "HEAD"(而不是 "GET" 或 "POST")作为第一个参数来调用 open() 方法,如 清单 9 所示。

    清单 9. 使用 Ajax 生成一个 HEAD 请求

       function getSalesData() {
         createRequest();
         var url = "/boards/servlet/UpdateBoardSales";
         request.open("HEAD", url, true);
         request.onreadystatechange = updatePage;
         request.send(null);
       }

    当您这样生成一个 HEAD 请求时,服务器并不会像对 GET 或 POST 请求一样返回一个真正的响应。相反,服务器只会返回资源的 头(header),这包括响应中内容最后修改的时间、请求资源是否存在和很多其他有用信息。您可以在服务器处理并返回资源之前使用这些信息来了解有关资源的信息。

    对于这种请求您可以做的最简单的事情就是简单地输出所有的响应头的内容。这可以让您了解通过 HEAD 请求可以使用什么。清单 10 提供了一个简单的回调函数,用来输出从 HEAD 请求中获得的响应头的内容。

    清单 10. 输出从 HEAD 请求中获得的响应头的内容

       function updatePage() {
         if (request.readyState == 4) {
           alert(request.getAllResponseHeaders());
         }
       }

    请参见 图 7,其中显示了从一个向服务器发出的 HEAD 请求的简单 Ajax 应用程序返回的响应头。

    您可以单独使用这些头(从服务器类型到内容类型)在 Ajax 应用程序中提供其他信息或功能。

    检查 URL

    您已经看到了当 URL 不存在时应该如何检查 404 错误。如果这变成一个常见的问题 —— 可能是缺少了一个特定的脚本或 servlet —— 那么您就可能会希望在生成完整的 GET 或 POST 请求之前来检查这个 URL。要实现这种功能,生成一个 HEAD 请求,然后在回调函数中检查 404 错误;清单 11 给出了一个简单的回调函数。

    清单 11. 检查某个 URL 是否存在

       function updatePage() {
         if (request.readyState == 4) {
           if (request.status == 200) {
             alert("URL exists");
           } else if (request.status == 404) {
             alert("URL does not exist.");
           } else {
             alert("Status is: " + request.status);
           }
         }
       }

    诚实地说,这段代码的价值并不太大。服务器必须对请求进行响应,并构造一个响应来填充内容长度的响应头,因此并不能节省任何处理时间。另外,这花费的时间与生成请求并使用 HEAD 请求来查看 URL 是否存在所需要的时间一样多,因为它要生成使用 GET 或 POST 的请求,而不仅仅是如 清单 7 所示一样来处理错误代码。不过,有时确切地了解目前什么可用也是非常有用的;您永远不会知道何时创造力就会迸发或者何时需要 HEAD 请求!

    有用的 HEAD 请求

    您会发现 HEAD 请求非常有用的一个领域是用来查看内容的长度或内容的类型。这样可以确定是否需要发回大量数据来处理请求,和服务器是否试图返回二进制数据,而不是 HTML、文本或 XML(在 JavaScript 中,这 3 种类型的数据都比二进制数据更容易处理)。

    在这些情况中,您只使用了适当的头名,并将其传递给 XMLHttpRequest 对象的 getResponseHeader() 方法。因此要获取响应的长度,只需要调用 request.getResponseHeader("Content-Length");。要获取内容类型,请使用 request.getResponseHeader("Content-Type");。

    在很多应用程序中,生成 HEAD 请求并没有增加任何功能,甚至可能会导致请求速度变慢(通过强制生成一个 HEAD 请求来获取有关响应的数据,然后在使用一个 GET 或 POST 请求来真正获取响应)。然而,在出现您不确定有关脚本或服务器端组件的情况时,使用 HEAD 请求可以获取一些基本的数据,而不需要对响应数据真正进行处理,也不需要大量的带宽来发送响应。

    结束语

    对于很多 Ajax 和 Web 程序员来说,本文中介绍的内容似乎是太高级了。生成 HEAD 请求的价值是什么呢?到底在什么情况下需要在 JavaScript 中显式地处理重定向状态代码呢?这些都是很好的问题;对于简单的应用程序来说,答案是这些高级技术的价值并不是非常大。

    然而,Web 已经不再是只需实现简单应用程序的地方了;用户已经变得更加高级,客户期望能够获得更好的稳定性、更高级的错误报告,如果应用程序有 1% 的时间停机,那么经理就可能会因此而被解雇。

    因此您的工作就不能仅仅局限于简单的应用程序了,而是需要更深入理解 XMLHttpRequest。

    ·如果您可以考虑各种就绪状态 —— 并且理解了这些就绪状态在不同浏览器之间的区别 —— 就可以快速调试应用程序了。您甚至可以基于就绪状态而开发一些创造性的功能,并向用户和客户回报请求的状态。
    ·如果您要对状态代码进行控制,就可以设置应用程序来处理脚本错误、非预期的响应以及边缘情况。结果是应用程序在所有的时间都可以正常工作,而不仅仅是只能一切都正常的情况下才能运行。
    ·增加这种生成 HEAD 请求的能力,检查某个 URL 是否存在,以及确认某个文件是否被修改过,这样就可以确保用户可以获得有效的页面,用户所看到的信息都是最新的,(最重要的是)让他们惊讶这个应用程序是如何健壮和通用。
    本文的目的并非是要让您的应用程序显得十分华丽,而是帮助您去掉黄色聚光灯后重点昭显文字的美丽,或者外观更像桌面一样。尽管这些都是 Ajax 的功能(在后续几篇文章中就会介绍),不过它们却像是蛋糕表面的一层奶油。如果您可以使用 Ajax 来构建一个坚实的基础,让应用程序可以很好地处理错误和问题,用户就会返回您的站点和应用程序。在接下来的文章中,我们将添加这种直观的技巧,这会让客户兴奋得发抖。(认真地说,您一定不希望错过下一篇文章!)

    相关文章

    最新评论