Get to know MDN better
此页面由社区从英文翻译而来。了解更多并加入 MDN Web Docs 社区。
sort() 方法就地对数组的元素进行排序,并返回对相同数组的引用。默认排序是将元素转换为字符串,然后按照它们的 UTF-16 码元值升序排序。
由于它取决于具体实现,因此无法保证排序的时间和空间复杂度。
如果想要不改变原数组的排序方法,可以使用 toSorted()。
定义排序顺序的函数。返回值应该是一个数字,其符号表示两个元素的相对顺序:如果 a 小于 b,返回值为负数,如果 a 大于 b,返回值为正数,如果两个元素相等,返回值为 0。NaN 被视为 0。该函数使用以下参数调用:
a第一个用于比较的元素。不会是 undefined。
b第二个用于比较的元素。不会是 undefined。
如果省略该函数,数组元素会被转换为字符串,然后根据每个字符的 Unicode 码位值进行排序。
经过排序的原始数组的引用。注意数组是就地排序的,不会进行复制。
如果没有提供 compareFn,所有非 undefined 的数组元素都会被转换为字符串,并按照 UTF-16 码元顺序比较字符串进行排序。例如“banana”会被排列到“cherry”之前。在数值排序中,9 出现在 80 之前,但因为数字会被转换为字符串,在 Unicode 顺序中“80”排在“9”之前。所有的 undefined 元素都会被排序到数组的末尾。
sort() 方法保留空槽。如果源数组是稀疏的,则空槽会被移动到数组的末尾,并始终排在所有 undefined 元素的后面。
备注:在 UTF-16 中,Unicode 字符超出 \uFFFF 的范围会被编码为两个代理码元(surrogate code unit),这些码位的范围是 \uD800 到 \uDFFF。每个码位的值都会被单独考虑进行比较。因此,由代理对 \uD855\uDE51 组成的字符将排在字符 \uFF3A 的前面。
如果提供了 compareFn,所有非 undefined 的数组元素都会按照比较函数的返回值进行排序(所有的 undefined 元素都会被排序到数组的末尾,并且不调用 compareFn)。
| > 0 | a 在 b 后,如 [b, a] |
| < 0 | a 在 b 前,如 [a, b] |
| === 0 | 保持 a 和 b 原来的顺序 |
所以,比较函数形式如下:
更正式地说,为了确保正确的排序行为,比较函数应具有以下属性:
符合上述限制的比较函数将始终能够返回 1、0 和 -1 中的任意一个,或者始终返回 0。例如,如果比较函数只返回 1 和 0,或者只返回 0 和 -1,它将无法可靠地排序,因为反对称性被破坏了。一个总是返回 0 的比较函数将不会改变数组,但仍然是可靠的。
默认的字典比较函数符合上述所有限制。
要比较数字而非字符串,比较函数可以简单的用 a 减 b,如下的函数将会将数组升序排列(如果它不包含 Infinity 和 NaN):
sort() 方法是通用的,它只期望 this 值具有 length 属性和整数键属性。虽然字符串也类似于数组,但此方法不适用于字符串,因为字符串是不可变的。
下述示例创建了四个数组,并展示原数组。之后对数组进行排序。对比了数字数组分别指定与不指定比较函数的结果。
对象数组可以通过比较它们的某个属性的值来排序。
当排序非 ASCII 字符的字符串(如包含类似 e、é、è、a、ä 等字符的字符串)。一些非英语语言的字符串需要使用 String.localeCompare。这个函数可以将函数排序到正确的顺序。
compareFn 可能会在数组中的每个元素上调用多次。根据 compareFn 的性质,这可能会产生很高的开销。如果 compareFn 执行的工作更多,需要排序的元素更多,使用 map() 进行排序可能更有效率。其思路是遍历数组一次,将用于排序的实际值提取到一个临时数组中,对临时数组进行排序,然后遍历临时数组以获得正确的顺序。
有一个开源库叫做 mapsort,它采用了这种方法。
sort() 方法返回对原始数组的引用,因此更改返回的数组将同时更改原始数组。
如果你希望 sort() 方法不会改变原始数组,而是返回一个类似于其他数组方法(如 map() )返回的浅拷贝数组,可以使用 toSorted() 方法。或者,你可以在调用 sort() 之前使用展开语法或 Array.from() 进行浅拷贝。
自 EcmaScript 第 10 版(EcmaScript 2019)起,规范 要求 Array.prototype.sort 为稳定排序。
假设有一个包含学生名字和年级的列表,已经将它按学生名字字母顺序进行预排序:
对这个数组执行 grade 升序排序后:
students 变量会具有以下值:
注意,那些年级相同的学生(如 Alex 和 Devlin)会维持调用排序之前的顺序,这是稳定排序所确保的。
EcmaScript 第 10 版(EcmaScript 2019)以前没有要求稳定性,意味着你可能会得到以下结果:
如果一个比较函数不符合纯函数、稳定性、自反性、反对称性和传递性规则,就像在描述中解释的那样,程序的行为是未定义的。
例如,请看这个示例:
在这个例子中,compareFn 函数是不规范的,因为它不满足反对称性:如果 a > b,它返回 1;但是通过交换 a 和 b,它返回了 0 而不是一个负值。因此,对于不同的引擎,结果数组也会有所不同。例如,V8(用于 Chrome、Node.js 等)和 JavaScriptCore(用于 Safari)根本不会对数组进行排序,而是返回 [3, 1, 4, 1, 5, 9];而 SpiderMonkey(用于 Firefox)将返回升序排序的数组 [1, 1, 3, 4, 5, 9]。
然而,如果 compareFn 函数稍微改变一下,使其返回 -1 或 0:
那么在 V8 和 JavaScriptCore 中,它将按降序排序,结果为 [9, 5, 4, 3, 1, 1],而 SpiderMonkey 返回的结果是原始数组:[3, 1, 4, 1, 5, 9]。
由于这种实现的不一致性,建议始终遵循五个约束条件以确保你的比较函数是规范的。
空槽会被移动到数组的末尾。
sort() 方法会读取 this 的 length 属性。然后它会收集在 0 到 length - 1 范围内所有已存在的整数键属性,对它们进行排序,然后写回。如果范围内存在缺失的属性,则相应的尾随属性将被删除,好像不存在的属性被排序到末尾一样。
| ECMAScript® 2027 Language Specification # sec-array.prototype.sort |
启用 JavaScript 以查看此浏览器兼容性表。