为什么应该使用库?
回到早期网络时代,就是 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?
有这么多优秀的方案供您选择,为什么选择 YUI 或 Dojo?一个词:完整性。不像其他方案需要附加的库或插件,Dojo 和 YUI 有现代前端工程师需要的一切(甚至更多)。这既有益处,也有害处,如果您是在市场上一次性购足 Ajax,那它们是两个强大的竞争者。除了大量的 Javascript 助手和工具之外,两者都提供一流的小部件和控件 — 远远超出标准浏览器有限的调板。
事实上,YUI 和 Dojo 是十分相似的。两者都提供一个健壮的核心实用工具和帮助函数,并且各自都有一个极好的小部件和组件的集合。除了日历、树、菜单这类旧备用组件之外,还有一个聊天选项。您可以选择一个强大的、充满生机的社区为您解答疑难、修复 bug、向其中增加新函数。他们有很多优秀的网站,其中有代码示例,参考资料,新手指南,以及 — 如果您喜欢纸质的 — 他们也提供书,书中有许多令人印象深刻的库。
Javascript 语言缺少名称空间或程序包,Dojo 和 YUI 克服了代码命名和包装的困难。在 YUI 中,代码位于 YAHOO 之下,附加一些 “包” 名。例如 YUI 的事件实用工具看起来像这样:
YAHOO.util.Event.addListener
YUI 3YUI 3 是一个比较新的、彻底重写的 YAHOO! 库。此更新为性能而设计,很有潜力。然而目前,小部件库还稍有欠缺,但是几个月内就可以得到改变。如果您今天 就开始,可能需要着重了解 YUI 2,但一定要关注未来的 YUI 3。
Dojo 遵循类似的方法,但它的实现简单一些。尽管小部件位于 dijit 之下,但您可以使用 Dojo 作为顶层包装器访问核心组件。
这两个库相似,但彼此并不完全相同。Dojo 允许您以声明 或编程 的方式使用它的小部件。换句话说,您可以使用您自己的标记为小部件配置特定属性,或者可以使用标准 Javascript 代码 — 可任意混合和匹配。Dojo 附带一个相当有用的依赖项管理器;当您需要一个特定模块时,Dojo 确保您具有所需要的一切。它也确保依赖项已经加载一次,且仅加载一次。
--------------------------------------------------------------------------------
Ajax 简单方法
尽管 XHR 对象自身不是很复杂,但这也有足够的细微差别使您很快就能理解当今 Ajax 库中的抽象层。Dojo 和 YUI 也不例外,它们都提供一个易于使用的包装器。当您使用一个库处理 XHR 时,能发现一些常见主题。至少,包装器应该:
提供被调用资源的 URI
提供传递参数的方式
提供一种指定被调用 HTTP 方法(get、post)的方法
包括一种处理调用结果的回调技术
YUI 和 Ajax
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 中。如您所见,您能检索某些元素 — 在本例中是一个
Dojo 和 Ajax
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 和 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 选择器
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 提供一组丰富的控件和小部件,使用它们能使您的应用程序更流行。手工输入日期很容易出错,多数用户期盼着能有某种控件。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 也有许多优秀的小部件 — 这些小部件被设计得可访问且支持国际化。在本例中,我们来看 Dojo 的声明式程序设计风格;不是手写几行 Javascript 代码,可以只是将下面这一行代码:
dojoType="dijit.form.DateTextBox"
添加到您想转变成日期选择器的字段。通过增加清单 6 的这两行代码,就能得到一个日历,几乎不费任何力气!
清单 6. Dojo 中的日期选择器
dojo.require("dojo.parser");
dojo.require("dijit.form.DateTextBox");
--------------------------------------------------------------------------------
获取帮助
无论您选择哪个库,有时您都可能需要一些帮助。YUI 和 Dojo 的特色是都有广泛的在线文档,从全面教程到详细的、循序渐进的例子。它们也有许多书供您选择,以便具有一个供探索的物理工件,它们都有充满活力的社区,乐于回答问题。最好的出发点就是它们的网站(参见 参考资料 中的链接)。
--------------------------------------------------------------------------------
结束语
本文在有限的篇幅内涵盖了很大的范围,坦白地说,这仅仅触及到这两个重要库的表面。如果您现在还没有使用某一个,我希望我能说服您试用 Dojo 或 YUI。每个都有充满热情的追随者,因此,都试一试,看它们怎么能帮助您向用户交付更好的体验。