【译】jQuery3新特性

jQuery 称霸 web 江湖已经有十年了,它屹立不倒是有原因的。jQuery 提供了用户容易操 作 DOM 的接口,执行 Ajax 请求,创建动画,以及更多。另外,与 DOM API 不同,jQuery 实现 了复合模式 (opens new window)。 因此,你可以使用在一个无论包含多少(零个、一个或者更多)元素的 jQuery 集合上调用 jQuery 方法。

在未来几周内,jQuery 随着第 3 个版本的发布将要达到一个重要的里程碑。jQuery3 修复 了很多 bug,增加了一些新的功能,弃用并删除了一些功能,并更改了一些功能的行为。在 这篇文章中,我将要突出介绍 jQuery3 最重要的变化。

# 新特性

以下各节中我将讨论在 jQuery3 中加入的重要特性。

# for...of循环

jQuery3 可能将使用 for...of 循环来遍历 jQuery 集合中的 DOM 元素。这种新的迭代器 是 ECMAScript2015(又名 ECMAScript 6)规范的一部分。这个新迭代器可以使你遍历迭代 对象(包括 Array,Map,Set,等等)。

当使用这个新的迭代器时,你在循环中接受到的值不是 jQuery 对象而是一个 DOM 元素。 这个迭代器将会略微的提高你操作 jQuery 集合的方式。

为了理解这个迭代器是怎样工作的,假设你想为页面上每一个input元素分配一个 ID。jQuery3 之前你会这么写:

var $inputs = $('input');

for (var i = 0; i < $inputs.length; i++) {
  $inputs[i].id = 'input-' + i;
}
1
2
3
4
5

在 jQuery3 中你可以这么写:

var $inputs = $('input');
var i = 0;

for (input of $inputs) {
  input.id = 'input-' + i++;
}
1
2
3
4
5
6

# $.get()$.post()增加了新的方法签名

为了$.get()$.post()的实用功能和$.ajax()一样,jQuery3 为其增加了新的方法 签名。添加的新的方法签名是:

$.get([settings]);

$.post([settings]);
1
2
3

settings是一个具有很多属性的对象。你可以提供给$.ajax()使用,它们是相同的对象 。想要了解更多关于$.ajax()的,请参 考$.ajax()网页 (opens new window)中的说明。

唯一的不同是,当 settings 对象传递给$.get()$.post()而不是$.ajax()时 ,method属性总是被忽略的。其原因是$.get()$.post()有一个预设的 HTTP 方法 来执行 Ajax 请求($.get()是 GET,$.post()是 POST)。基本上,你不能通 过$.get()发送一个 POST 请求的。

思考下面的代码

$.get({
  url: 'https://www.audero.it',
  method: 'POST', // 这个属性会被忽略
});
1
2
3
4

尽管设置了method属性,这段代码是不会发送一个 POST 请求,而是一个 GET 请求。

# 动画使用requestAnimationFrame()

所有的现在浏览器,包括 IE10 及以上, 都支持 requestAnimationFrame (opens new window)。jQuery3 将会在内部使用这个 API 执行动画,使动画更加流畅并且减少对 CPU 资源的消耗。

# unwrap()

jQuery3 给unwrap()添加了一个可选参数,这个新的方法签名是:

unwrap([selector]);
1

感谢这个变化,您将能够通过选择器表达式的字符串在已经选择的父元素中去匹配。如果存 在匹配项,则匹配的子元素的父元素被移除;否则不执行操作。

# 改变的特性

jQuery3 也修改了一些特性的行为。

# :visible:hidden

新版本的 jQuery 修改了:visible:hidden过滤器的释义。如果元素有任何的布局容 器,即使宽度和(或)高度都为 0,也将会视为:visible。举例说明,br元素和没有内 容的行级元素(内联元素)现在都会被:visible过滤器选中。

因此,如果在一个页面中有以下标记语言:

<div></div>
<br />
1
2

然后你运行以下语句:

console.log($('body :visible').length);
1

在 jQuery1.x 和 jQuery2.x 中你得到的结果为 0,但是在 jQuery3 中你得到的结果是 2。

# data()

另一个重要的变化是有关data()方法的行为。他已经变得 和Dataset API (opens new window)规范一样了 。jQuery 3 将所有属性的键都转换成驼峰式命名方式。要想理解这个变化,请思考下面的 元素:

<div id="container"></div>
1

如果你使用的 jQuery3 之前的版本你会这么写:

var $elem = $('#container');
$elem.data({
  'my-property': 'hello',
});
console.log($elem.data());
1
2
3
4
5

在控制台你将会得到如下的结果:

{my-property:"hello"}
1

而使用 jQuery3 你将得到如下的结果:

{myPropery:"hello"}
1

请注意,在 jQuery3 中属性名已经变成了驼峰式没有横杠(-),而在以前的版本中,属性 名会保持全小写和保留横杠(-)。

# Deferred对象

jQuery3 改变了Deferred对象的行为,Deferred对象是Promise对象的前身,这次改 变提高了Deferred对象对Promise/A+提案 (opens new window)的兼容性 。Deferred这个对象及其它的历史都是很有趣的。想要了解更多,你可以去阅 读官方文档 (opens new window)或者阅读我写的 书jQuery 实战(第三版) (opens new window), 这本书也包含了 jQuery3。

在 jQuery1.x 和 2.x 中,传递给Deferred的回调函数内出现未捕获的异常会阻断程序的 执行。不像原生Promise对象那样会抛出异常冒泡至window.onerror(通常冒泡到这里 )。如果你没有定义一个函数处理错误事件(通常我们是会处理的),那么异常信息就会显 示并且程序会终止执行。

jQuery3 遵循原生Promise对象的模式。因此,抛出的异常被当作失败,接着失败回调函 数执行。一旦失败回调函数执行完成,进程就会继续,下面的的成功回调函数将被执行。

为了帮助你理解两者不同,让我们看一个小例子。思考下面的代码:

var deferred = $.Deferred();

deferred
  .then(function () {
    throw new Error('An error');
  })
  .then(
    function () {
      console.log('Success 1');
    },
    function () {
      console.log('Failure 1');
    }
  )
  .then(
    function () {
      console.log('Success 2');
    },
    function () {
      console.log('Failure 2');
    }
  );

deferred.resolve();
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

在 jQuery1 和 jQuery2 中,只有第一个函数(抛出错误的函数)被执行。另外,因为我没 有为window.onerror定义任何处理函数,所以控制台将会输出“Uncaught Error: An error”并且程序执行将会终止。

在 jQuery3 中,行为是完全不同的。你将在控制台上看到“Failure 1”和“Success 2”两条 信息。异常被第一个失败回调函数处理,并且一旦被处理,随后的成功回调函数也会被执行 。

# SVG 文档

没有任何版本的 jQuery,包括 jQuery3,官方正式支持 SVG 文档。然而,事实是很多方法 是可以工作的,还有另外一些,比如类名操作在以前是不可正常使用的,但是它们在 jQuery3 中更新后也可以正常使用了。因此,在即将到来的 jQuery 版本中,你可以放心地 在 SVG 文档上使用addClass()hasClass()方法。

# 已废弃或者已移除的方法和属性

除目前所描述的改进外,jQuery 还移除和废弃了一些特性。

# 废弃bind()unbind()delegate()undelegate()

很久以前(jQuery1.7 版本时)jQuery 就引入了on()方法,它提供了统一的接口替 换bind()delegate()live()方法。与此同时 jQuery 也提供了off()方法,同 样提供了统一的接口替换unbind()undelegated()die()方法。 从on()off()方法引入时就不推荐使 用bind()delegate()unbind()undelegate()方法,但它们还是一直存在着。

jQuery3 废弃了这些方法并且打算在未来的版本(可能是 jQuery4)会移除它们。在你所有 的项目中坚持使用on()off()方法,这样你就不必担心未来的版本更新。

# 移除load()unload()error()方法

jQuery3 抛弃了已经废弃的load()unload()error()方法。这些方法很久以前( 在 jQuery1.8 时)就被废弃了,但是他们仍旧留在 jQuery 中。如果你使用的插件依赖其 中的一个或多个方法,升级到 jQuery3 后你的代码将会崩溃。因此,在升级时注意。

# 移除contextsupportselector属性

jQuery3 抛弃了已经废弃 的context (opens new window)support (opens new window)selector (opens new window)属 性。正如我在上一节讲到的,如果你仍旧在项目中使用这些属性或者使用的插件依赖这些属 性,升级到 jQuery3 后你的代码将会崩溃。

# Bug 修复

jQuery3 修复了在以前版本中存在的重大 bug。在下面章节中,我将介绍两点,这两点在你 的工作中会有很大的不同。

# width()height()的值不再四舍五入

jQuery3 修复了width()height()及所有其它相关方法的 bug。这些方法将不再将结 果像素值四舍五入到整数值,因为四舍五入后在某些情况下很难对元素进行定位。

为了理解这个问题,假设你有一个宽度为 100px 的容器元素,它里面包含三个宽度为三分 之一(33.333333%)的元素:

<div class="container">
    <div>My name</div>
    <div>is</div>
    <div>Aurelio De Rosa</div>
</div
1
2
3
4
5

在 jQuery3 之前,如果你尝试用以下代码来获取子元素的宽度……

$('.container div').width();
1

……你得到的结果是 33。原因是 jQuery 将 33.33333 四舍五入了。在 jQuery3 中,这个 bug 被修复了,因此你得到结果是更加精确的(即浮点数的结果)。

# wrapAll()

新版本 jQuery 修复了一个当传递函数给wrapAll()时发生的 bug。在 jQuery 之前,当 传递一个函数给wrapAll()时,jQuery 集合中每一个元素都被包裹。换句话说,这种行为 和传递一个函数给wrap()是一样的。

除了修复这个问题,还有一个变化:由于这个函数在 jQuery3 中只会被调用一次,因此 jQuery 集合每个元素的索引不能传入这个方法中。最终这个方法的上下文(this)将指 向 jQuery 集合中的第一个元素。

# 下载 jQuery3 beta1

如果你对这篇文章感兴趣,你可能想要尝 试jQuery3 第一个 beta 版 (opens new window)。 您可以通过访问下面两个 url 来获得。

它在 npm 上也是可用的,你可以通过运行下面的命令来下载:

npm install jquery@3.0.0-beta1
1

# 总结

很多人认为 jQuery 将会死掉,认为在现代网页开发中已经没有一席之地。然而其仍在持续 发展,jQuery 使用统计(在排名前 100 万的网站中使用率为 78.5%)反驳了这些说法。

在这篇文章中,我已经带着你介绍了 jQuery3 将会带来的重要特性变化。或许你可能已经 注意到了,这个版本是不太会对你现有的项目造成崩溃,因为它没有引入一些重大的变化。 尽管如此,在升级期间还是有一些事项需要注意得,如Deferred对象的改进。在更新第三 方插件时也要注意,审查项目将帮助你发现异常行为或者崩溃的功能。

【译】ECMAScript 6中的参数使用

【译】ECMAScript 6中的参数使用

使用Sublime Text打造Web集成开发环境

使用Sublime Text打造Web集成开发环境