Ajax 库大比拼:Dojo 与 YUI
使用库创建更好的用户体验
简介: 从当地报纸到 CEO 们冲浪的网站,异步 JavaScript 和 XML(Ajax)随处可见。与现在流行的观点相反,它不是什么高科技,只不过就是使用适当的库。探索目前流行的 Yahoo!User Interface(YUI) 和 Dojo 库,学习它们如何简化典型的 Ajax 技术,以及使 JavaScript 更容易使用。弄清楚为什么应该一开始就使用库,以及如何选择库。也提供了一些 YUI 和 Dojo 的具体例子。
创建一个现代的基于 Web 的应用程序比起几年前所需更多。现在,开发人员的任务是利用 Ajax 工具箱中所有技术来创建丰富的交互式 UI。这些方法最初引入时,许多专家认为只有业内行家才能掌握,尽管现在并没有变简单,但是高质量的框架已经可以帮您的大忙了。本文探讨 Dojo 和 Yahoo! UI Library (YUI) 这类流行的库如何简化当今 JavaScript 开发人员的工作,为什么应该一开始就使用库,以及如何选择库。也提供了一些 Dojo 和 YUI 的具体例子。
回到早期网络时代,就是 20 世纪 90 年代后期,我们这些做客户端开发的人面前困难重重。比起今天时髦的智能电话,我们的显示器分辨率很低;浏览器大战如火如荼,带给我们的是各种各样的不兼容;我们的工具箱只比那些使用穿孔卡片的人好一点点,我们艰难地进行着工作。使 JavaScript 跨浏览器正常工作是一项艰巨的任务,在早期的项目中,我们遇到了很多麻烦,管理员规定 “只能使用让应用程序运转所需的最小限度的 JavaScript 代码”。那段日子不堪回首。
问题已经得到了妥善的处理,但是浏览器问题到今天仍然存在(Windows® Internet Explorer® 6 ,我将目光投向你)。今天,除了改进很大的工具箱外,您还可以利用大量的库,来减轻开发跨浏览器前端代码的压力。事实上,您总是面临一个 选择的悖论,选择正好合适的框架使您轻松很多 — 但是,您不得不花大量时间在 Y 操作系统上调试浏览器 X 的这种间歇性问题。现在,您应该关注使用库为您的编码工作建立基础 — 在各种浏览器和您所写的代码之间有不同的虚拟机。
除了消除浏览器之间的差别外,在您的日常编码工作中,工具包也会给您额外的帮助。有些库是极简的,仅仅只做一件事,而有些库则像包罗万象的厨房,包含所有型号(sink-sized)的工具箱,试图改正 Web 中的所有缺陷。尽管每个库都是不一样的,但仍然有一些共同点。大部分提供:
- 一个包装
XMLHttpRequest
(XHR) 对象的事务包装器 - 跨浏览器 CSS 选择符
- 简化的事件处理
- 各种各样的动画、效果和小部件
- 各种应用函数
您怎么选择这个库,而不选另一个?唯一错误的回答是手工编写代码 — 不使用任何 库。选择库可能很困难,但是比起自己手工编写代码来说,就无足轻重了。在您为必须选择一个库而慌乱之前,要知道它们大多数都可以很好地协同工作(合理范围内)。有时候,混合和匹配是最好的选择。不管怎么样,以下建议供您选择库时参考:
- 您想从中得到什么?您想要寻找一个几乎可以完全替代页面上所有 UI 元素的替代品?或者您正在寻找某种方式来简化 JavaScript 编程?
- 代码有多容易阅读?尽管过去几年中文档有了很大的改进,但是有时您也必须深入钻研代码。托付给一个库之前,有必要花点时间研究一下源代码。它容易理解吗?或者,甚至原作者也在此遇到了麻烦?
- 文档有多好?清晰易读的代码能弥补文档的不足,但是除了教程和例子,没有什么能够帮助您开始。随便看看维基百科和网站,查看他们提供了什么。例子是清晰易读的吗?Google 的快速搜索能带给您合适的资料吗?
- 此库相关的社区是什么样的?查看邮件列表,是否有很多邮件?新人受到尊重还是嘲笑?代码近期有更新,或者还是多年前的版本?
- 您能得到帮助吗?尽管这涉及到社区早期内容,但浏览社区看看人们正在用什么,总是有价值的。浏览求职板块,对简历中频繁出现的库有一个大概了解。
做最终决定之前,一定要花一点时间考虑备选库。一些库有自己的风格 — 比如,Prototype 会为 JavaScrip 编程带入大量的 Ruby。如果您认为 Ruby 是所有语言的终结,那么这就是它的一个特征。但是如果阅读使您的眼睛不舒服,Prototype 可能就不是最好的选择。跟着教程阅读代码是不错,但是,在您用一个给定库实际解决自己 的问题时,很难知道哪个适合您。
有这么多优秀的方案供您选择,为什么选择 YUI 或 Dojo?一个词:完整性。不像其他方案需要附加的库或插件,Dojo 和 YUI 有现代前端工程师需要的一切(甚至更多)。这既有益处,也有害处,如果您是在市场上一次性购足 Ajax,那它们是两个强大的竞争者。除了大量的 JavaScript 助手和工具之外,两者都提供一流的小部件和控件 — 远远超出标准浏览器有限的调板。
事实上,YUI 和 Dojo 是十分相似的。两者都提供一个健壮的核心实用工具和帮助函数,并且各自都有一个极好的小部件和组件的集合。除了日历、树、菜单 这类旧备用组件之外,还有一个聊天选项。您可以选择一个强大的、充满生机的社区为您解答疑难、修复 bug、向其中增加新函数。他们有很多优秀的网站,其中有代码示例,参考资料,新手指南,以及 — 如果您喜欢纸质的 — 他们也提供书,书中有许多令人印象深刻的库。
JavaScript 语言缺少名称空间或程序包,Dojo 和 YUI 克服了代码命名和包装的困难。在 YUI 中,代码位于 YAHOO 之下,附加一些 “包” 名。例如 YUI 的事件实用工具看起来像这样:
YAHOO.util.Event.addListener |
Dojo 遵循类似的方法,但它的实现简单一些。尽管小部件位于 dijit 之下,但您可以使用 Dojo 作为顶层包装器访问核心组件。
这两个库相似,但彼此并不完全相同。Dojo 允许您以声明 或编程 的方式使用它的小部件。换句话说,您可以使用您自己的标记为小部件配置特定属性,或者可以使用标准 JavaScript 代码 — 可任意混合和匹配。Dojo 附带一个相当有用的依赖项管理器;当您需要一个特定模块时,Dojo 确保您具有所需要的一切。它也确保依赖项已经加载一次,且仅加载一次。
尽管 XHR 对象自身不是很复杂,但这也有足够的细微差别使您很快就能理解当今 Ajax 库中的抽象层。Dojo 和 YUI 也不例外,它们都提供一个易于使用的包装器。当您使用一个库处理 XHR 时,能发现一些常见主题。至少,包装器应该:
- 提供被调用资源的 URI
- 提供传递参数的方式
- 提供一种指定被调用 HTTP 方法(
get、post
)的方法 - 包括一种处理调用结果的回调技术
YUI 将其 XHR 包装器隐藏在连接管理器中。API 很直观:您调用单个方法,并返回一个连接对象。一个典型的调用如清单 1 所示。
清单 1. 一个 YUI Ajax 调用样例
var url = "/fooApp/validate"; var msg_div = document.getElementById("messages"); var callback = { success: function(o) { msg_div.innerHTML = o.responseText}, failure: function(o) { console.debug("An error occurred: ", o);} }; function validate() { var form = document.getElementById("pim"); YAHOO.util.Connect.setForm(form); var transaction = YAHOO.util.Connect.asyncRequest("GET", url, callback); } |
比起其他库,这看起来有点冗长,这是因为,为了便于阅读我展开了一些。您可以将回调函数和 URL 直接放入调用函数Connect.asyncRequest
中。如您所见,您能检索某些元素 — 在本例中是一个 <div>
,它将容纳服务器返回的信息 — 以及包含您想要发送到服务器的数据的表单。在连接管理器上调用 setForm
方法,会基于您所做的调用类型(在本例中是 GET
请求),收集表单并将它们适当地打包。当验证方法被调用时,对服务器的异步调用将被触发。
Dojo 的方法类似于 YUI 方法,只不过没那么冗长。Dojo 的 XHR 包装器在 Dojo Base 内,并且提供针对几个主要 HTTP 动词(post、get、put 和 delete
)的方法。清单 2 中有一种调用 Dojo 方法的方式。
清单 2. 一个 Dojo Ajax 调用样例
var xhrParms = { url: "/fooApp/validate", load: function(response){ dojo.byId("messages").innerHTML = response; }, error: function(data){ console.debug("An error occurred: ", data); }, timeout: 2000, form: "pim" }; function validate() { dojo.xhrGet(xhrParms); } |
我再次选择展开代码来提高可读性,但是,如果您喜欢,您可以将参数直接放在调用函数 dojo.xhrGet
中。正如您所见,这个例子和 YUI 有点相似,但有一些很有趣的差异。第一,您可能注意到明显缺乏 document.getElementById
调用。通过 ID 获取元素是常用的方法,但是超过 20 个字符时很容易发生输入错误!很多库提供各种形式的快捷方式,Dojo 提供 dojo.byId
。尽管不如 Prototype 的 $
这样精炼,但仍然是一个很大的改进。同样,当 validate
方法被调用时,您对服务器的调用将被调用。
在 web 开发人员的工具箱中,CSS 选择器是一个功能强大的工具,但是跟 JavaScript 一样,浏览器支持不具有普遍性。幸好,现在的库填补了这个空缺,给我们提供 CSS 选择器的全部功能。
YUI 利用其 Selector
实用工具为您提供 CSS 3 选择器的全部功能。本质上就是,您在一个特定的CSS 选择器上创建一个 “查询” ,就能够得到一个匹配该条件的元素集。例如,见清单 3。
清单 3. 利用 YUI 实现的 CSS 选择器
YAHOO.util.Event.onDOMReady(bindEvents); function bindEvents() { var headers = YAHOO.util.Selector.query('.header'); YAHOO.util.Event.on(headers, 'click', toggleSection); } |
在本例中,您是在寻找具有报头样式的元素,并对返回的每个元素应用一个事件侦听器(更多介绍见下节)。可以看出来,这是一个简洁的、功能强大的元素检索方法,在此代码中,每当有人单击一个 header 样式的元素时,toggleSection
方法就将被调用。
Dojo 有一个功能同样强大的选择器机制供其使用。它也调用它的查询 实现,同上面一样,您给它一个 CSS 选择器,它就返回一个元素集(如清单 4 所示)。
清单 4. 利用 Dojo 实现的 CSS 选择器
dojo.addOnLoad(bindEvents); function bindEvents() { dojo.query('.header').forEach( function(header) { dojo.connect(header, "click", toggleSection); }); } |
收到每个带有 header 样式的元素后,用 forEach
方法遍历集合,对每个元素应用一个事件侦听器。
在 unobtrusive JavaScript 时代,事件处理尤为常见。现代的浏览器不总是能完成任务,但是库设计者再次独自揽下重任,以简化我们的生活。在前一节,您已经看见两种绑定到单击事件的方法:YUI 的 Event
和 Dojo 的 connect
。在这两个例子中,您要求事件发生时能得到通知,以便采取措施 — 在上面例子中是进行切换(也就是说,如果是可见的,则隐藏,如果是隐藏的,则使其可见)。您可以直接向您的标记添加事件处理程序,但是跟您在这里看到的一样使用抽象则更为简洁,因为它将您的业务逻辑从表示中分离出来,使标记更简洁、代码更容易理解。
您不能将事件附加到元素上,除非该元素已加载 — 但是您如何知道这什么时候发生?Dojo 和 YUI 为您提供一个帮助方法。在 YUI 中,我们有:
YAHOO.util.Event.onDOMReady() |
Dojo 为您提供:
dojo.addOnLoad() |
不管哪种情况,您都可以利用这些帮助工具,来确保文档就绪时附加事件。完成该任务还有一些其他的方法,但是没有几种比这两种方法更容易理解。
现代网页设计的调板很单调:只有文本框、文本区、按钮等,不是很多。再次,当浏览器让您失望时,库设计者让您恢复信心。现在,您几乎具有了富桌面应用程序中可用的每一样东西:有日期选择器、菜单、树、滑块,窗口等等。这是一个真正的聚宝盆!让我们看看 YUI 和 Dojo 中的日期选择器。
YUI 提供一组丰富的控件和小部件,使用它们能使您的应用程序更流行。手工输入日期很容易出错,多数用户期盼着能有某种控件。YUI 有一个很棒的日期选择器,位于 Calendar
组件中(如清单 5 所示)。
清单 5. YUI 中的日期选择器
var cal = new YAHOO.widget.Calendar("cal", "cal1Container", {navigator:true}); var bday = document.getElementById("bday"); YAHOO.util.Event.addListener(bday, "focus", renderCal); function handleSelect(type,args,obj) { var dates = args[0]; var date = dates[0]; var year = date[0], month = date[1], day = date[2]; bday.value = month + "/" + day + "/" + year; cal.hide(); } function renderCal() { cal.selectEvent.subscribe(handleSelect, cal, true); cal.render(); cal.show(); } |
这有许多函数,但是其中很多您以前都见过。调用 YAHOO.widget.Calendar
创建一个日历小部件,并在 birthday 字段上建立一个监听器,这使得在接收到焦点时,日历出现。handleSelect
函数将用户选择的值输入到 birthday 字段。这只是冰山一角。YUI 日历很容易配置,它也支持国际化。
Dojo 也有许多优秀的小部件 — 这些小部件被设计得可访问且支持国际化。在本例中,我们来看 Dojo 的声明式程序设计风格;不是手写几行 JavaScript 代码,可以只是将下面这一行代码:
dojoType="dijit.form.DateTextBox" |
添加到您想转变成日期选择器的字段。通过增加清单 6 的这两行代码,就能得到一个日历,几乎不费任何力气!
清单 6. Dojo 中的日期选择器
dojo.require("dojo.parser"); dojo.require("dijit.form.DateTextBox"); |
无论您选择哪个库,有时您都可能需要一些帮助。YUI 和 Dojo 的特色是都有广泛的在线文档,从全面教程到详细的、循序渐进的例子。它们也有许多书供您选择,以便具有一个供探索的物理工件,它们都有充满活力的社区,乐于回答问题。最好的出发点就是它们的网站(参见 参考资料 中的链接)。
本文在有限的篇幅内涵盖了很大的范围,坦白地说,这仅仅触及到这两个重要库的表面。如果您现在还没有使用某一个,我希望我能说服您试用 Dojo 或 YUI。每个都有充满热情的追随者,因此,都试一试,看它们怎么能帮助您向用户交付更好的体验。
学习
- Ajax:交易工具:发现更多的 JavaScript 开发人员工具。
- “用 Dojo 1.x 开发小部件”(developerWorks,2009 年 4 月):了解使用 Dojo JavaScript 工具箱开发 HTML 小部件的基础知识。
- “用 Dojo 和 Ajax 创建可重用和可重新发布的组件”(developerWorks,2008 年 6 月):使用 Dojo 和 Ajax 开发可以与核心应用程序轻松集成的可重用组件。
- “编写更优美的网页”(developerWorks,2009 年 6 月):介绍两个用于网页构建的框架:Blueprint 和 Yahoo!® User Interface (YUI) Grid。
XMLHttpRequest
:了解关于 XHR 对象的更多信息。
- Behavioral Separation:了解更多关于在网页设计和编程中分离内容、样式和行为的好处。
- YUI Library:了解关于 Yahoo! UI Library 的更多信息。
- Dojo:了解关于 Dojo Toolkit 的更多信息。
- developerWorks Web development 专区:通过专门关于 Web 技术的文章和教程,扩展您在网站开发方面的技能。
- developerWorks Ajax 资源中心:这是有关 Ajax 编程模型信息的一站式中心,包括很多文档、教程、论坛、blog、wiki 和新闻。任何 Ajax 的新信息都能在这里找到。
- developerWorks Web 2.0 资源中心,这是有关 Web 2.0 相关信息的一站式中心,包括大量 Web 2.0 技术文章、教程、下载和相关技术资源。您还可以通过 Web 2.0 新手入门 栏目,迅速了解 Web 2.0 的相关概念。
- 查看 HTML5 专题,了解更多和 HTML5 相关的知识和动向。
讨论
没有评论:
发表评论