如果我有一个 JavaScript 对象,例如:
var list = {
"you": 100,
"me": 75,
"foo": 116,
"bar": 15
};
有没有一种方法可以根据值对属性进行排序?这样我最后会得到:
list = {
"bar": 15,
"me": 75,
"you": 100,
"foo": 116
};
如果我有一个 JavaScript 对象,例如:
var list = {
"you": 100,
"me": 75,
"foo": 116,
"bar": 15
};
有没有一种方法可以根据值对属性进行排序?这样我最后会得到:
list = {
"bar": 15,
"me": 75,
"you": 100,
"foo": 116
};
let maxSpeed = {
car: 300,
bike: 60,
motorbike: 200,
airplane: 1000,
helicopter: 400,
rocket: 8 * 60 * 60
};
let sortable = [];
for (var vehicle in maxSpeed) {
sortable.push([vehicle, maxSpeed[vehicle]]);
}
sortable.sort(function(a, b) {
return a[1] - b[1];
});
// [["bike", 60], ["motorbike", 200], ["car", 300],
// ["helicopter", 400], ["airplane", 1000], ["rocket", 28800]]
let objSorted = {}
sortable.forEach(function(item){
objSorted[item[0]]=item[1]
})
Object.entries()
将对象转换为数组:
const maxSpeed = {
car: 300,
bike: 60,
motorbike: 200,
airplane: 1000,
helicopter: 400,
rocket: 8 * 60 * 60
};
const sortable = Object.entries(maxSpeed)
.sort(([,a],[,b]) => a-b)
.reduce((r, [k, v]) => ({ ...r, [k]: v }), {});
console.log(sortable);
Object.fromEntries()
将数组转换为对象。然后代码可以简化为这样:
const maxSpeed = {
car: 300,
bike: 60,
motorbike: 200,
airplane: 1000,
helicopter: 400,
rocket: 8 * 60 * 60
};
const sortable = Object.fromEntries(
Object.entries(maxSpeed).sort(([,a],[,b]) => a-b)
);
console.log(sortable);
_.pairs
将一个对象转换为 [ [key1, value1], [key2, value2] ] 的形式。然后对该数组调用 sort
方法进行排序,然后再使用 _.object
将其转回对象形式。 - dansch我们不希望复制整个数据结构,也不想使用需要关联数组的数组。
这里有另一种执行与Bonna相同操作的方法:
var list = {"you": 100, "me": 75, "foo": 116, "bar": 15};
keysSorted = Object.keys(list).sort(function(a,b){return list[a]-list[b]})
console.log(keysSorted); // bar,me,you,foo
keysSorted
是一个数组!不是对象! - Green.map(key => list[key]);
,它将返回整个对象而不仅仅是键。 - James Moran您的对象可以有任意数量的属性,如果将对象放入数组中,则可以选择按任何对象属性进行排序,无论是数字还是字符串。考虑以下数组:
var arrayOfObjects = [
{
name: 'Diana',
born: 1373925600000, // Mon, Jul 15 2013
num: 4,
sex: 'female'
},
{
name: 'Beyonce',
born: 1366832953000, // Wed, Apr 24 2013
num: 2,
sex: 'female'
},
{
name: 'Albert',
born: 1370288700000, // Mon, Jun 3 2013
num: 3,
sex: 'male'
},
{
name: 'Doris',
born: 1354412087000, // Sat, Dec 1 2012
num: 1,
sex: 'female'
}
];
按出生日期排序,最早的优先。// use slice() to copy the array and not just make a reference
var byDate = arrayOfObjects.slice(0);
byDate.sort(function(a,b) {
return a.born - b.born;
});
console.log('by date:');
console.log(byDate);
按名称排序
var byName = arrayOfObjects.slice(0);
byName.sort(function(a,b) {
var x = a.name.toLowerCase();
var y = b.name.toLowerCase();
return x < y ? -1 : x > y ? 1 : 0;
});
console.log('by name:');
console.log(byName);
substr
去掉第一个字符;否则 Diana
和 Debra
将没有定义的顺序。此外,你的 byDate
排序实际上使用的是 num
而不是 born
。 - Lawrence Dolx.localeCompare(y)
。 - Jemar JoneslocalCompare
看起来是一个非常有用的函数!只需要注意它不会在每个浏览器中得到支持 - IE 10及以下版本,Safari Mobile 9及以下版本。 - inorganikECMAScript 2017 引入了 Object.values / Object.entries
。顾名思义,前者将对象的所有值聚合成一个数组,后者将整个对象转换为一个 [键, 值]
数组的数组;类似于 Python 的dict.values()
和dict.items()
。
这些特性使得将哈希排序为有序对象变得更加容易。目前只有少部分 JavaScript 平台支持它们,但你可以在 Firefox 47+ 上尝试。
编辑:现在被 所有现代浏览器支持!
let obj = {"you": 100, "me": 75, "foo": 116, "bar": 15};
let entries = Object.entries(obj);
// [["you",100],["me",75],["foo",116],["bar",15]]
let sorted = entries.sort((a, b) => a[1] - b[1]);
// [["bar",15],["me",75],["you",100],["foo",116]]
let sorted = Object.entries(obj).sort((a, b) => a[1] - b[1]);
- Avid为了完整起见,该函数返回对象属性的排序数组:
function sortObject(obj) {
var arr = [];
for (var prop in obj) {
if (obj.hasOwnProperty(prop)) {
arr.push({
'key': prop,
'value': obj[prop]
});
}
}
arr.sort(function(a, b) { return a.value - b.value; });
//arr.sort(function(a, b) { return a.value.toLowerCase().localeCompare(b.value.toLowerCase()); }); //use this to sort as strings
return arr; // returns array
}
var list = {"you": 100, "me": 75, "foo": 116, "bar": 15};
var arr = sortObject(list);
console.log(arr); // [{key:"bar", value:15}, {key:"me", value:75}, {key:"you", value:100}, {key:"foo", value:116}]
这里是包含上述代码的JSFiddle。这个解决方案基于这篇文章。
这里是用于字符串排序的更新版fiddle。你可以从中删除所有额外的.toLowerCase()
转换,以进行区分大小写的字符串比较。
参考 @marcusR 的回答的箭头版本
var myObj = { you: 100, me: 75, foo: 116, bar: 15 };
keysSorted = Object.keys(myObj).sort((a, b) => myObj[a] - myObj[b]);
alert(keysSorted); // bar,me,you,foo
UPDATE: April 2017
This returns a sorted myObj
object defined above.
const myObj = { you: 100, me: 75, foo: 116, bar: 15 };
const result =
Object.keys(myObj)
.sort((a, b) => myObj[a] - myObj[b])
.reduce(
(_sortedObj, key) => ({
..._sortedObj,
[key]: myObj[key]
}),
{}
);
document.write(JSON.stringify(result));
UPDATE: March 2021 - Object.entries with sort function (updated as per comments)
const myObj = { you: 100, me: 75, foo: 116, bar: 15 };
const result = Object
.entries(myObj)
.sort((a, b) => a[1] - b[1])
.reduce((_sortedObj, [k,v]) => ({
..._sortedObj,
[k]: v
}), {})
document.write(JSON.stringify(result));
var myObj = {"1": {"Value": 40}, "2": {"Value": 10}, "3": {"Value": 30}, "4": {"Value": 20}};
运行。 - Rohanilentries
。根据标准,一个对象是属性的无序集合。这意味着,如果您尝试通过属性值或其他任何方式对新对象进行排序构造,属性键的顺序再次变得未定义。例如,Chrome默认按属性键排序,因此每次尝试以不同的方式对它们进行排序都是无用的。您唯一的机会是根据您的排序偏好获取“索引”,并相应地遍历原始对象。 - ZorgoZObject.assign()
比在reduce中展开_sortedObj
要更高效。 - DiegoRBaqueroJavaScript对象在定义上是无序的(详见ECMAScript语言规范第8.6节)。甚至语言规范都不能保证如果你连续两次迭代同一个对象的属性,它们第二次出现的顺序会与第一次相同。
如果您需要有序的东西,请使用数组和Array.prototype.sort方法。
好的,可能你已经知道,JavaScript有一个sort()函数可以用来对数组进行排序,但是没有类似的函数可以用来对对象进行排序...
所以在这种情况下,我们需要以某种方式获取键的数组并对它们进行排序,这也是为什么大多数时候API会把对象放在数组中返回,因为Array比对象字面量拥有更多与之配合的原生函数,无论如何,快速的解决方案是使用Object.key函数,它返回一个对象的键的数组。我编写了下面的ES6函数来帮助你完成这个任务,它使用JavaScript中的本地sort()和reduce()函数:
function sortObject(obj) {
return Object.keys(obj)
.sort().reduce((a, v) => {
a[v] = obj[v];
return a; }, {});
}
现在你可以像这样使用它:
let myObject = {a: 1, c: 3, e: 5, b: 2, d: 4};
let sortedMyObject = sortObject(myObject);
查看sortedMyObject,您可以看到按键排序的结果如下:
{a: 1, b: 2, c: 3, d: 4, e: 5}
同样的方式,主要对象不会被更改并且我们实际上得到了一个新的对象。
我还创建了下面这张图片,以使函数步骤更加清晰,以防您需要稍微更改一下来适应您的需求:
c: 3
更改为 c: 13
,您会看到它崩溃了。 - VtoCorleone使用ES6进行更新:如果您的关注点是拥有一个排序后的对象以便迭代(这就是我想像您需要排序对象属性的原因),则可以使用Map对象。
您可以按排序顺序插入键值对,然后使用for..of
循环将保证按照插入顺序进行迭代。
var myMap = new Map();
myMap.set(0, "zero");
myMap.set(1, "one");
for (var [key, value] of myMap) {
console.log(key + " = " + value);
}
// 0 = zero
// 1 = one
for-in
或Object.keys
所遵循,但它被定义为Object.getOwnPropertyNames
和其他新方法用于访问属性名称数组时所遵循的顺序。因此,这是另一个选项。 - T.J. CrowderMap
实际上可以拥有一个可以按排序顺序存储数据的数据结构(对数据进行排序,循环遍历并将其存储在 Map 中),而使用对象则不能保证这一点。 - julianljkvar list = {
"you": 100,
"me": 75,
"foo": 116,
"bar": 15
};
function sortAssocObject(list) {
var sortable = [];
for (var key in list) {
sortable.push([key, list[key]]);
}
// [["you",100],["me",75],["foo",116],["bar",15]]
sortable.sort(function(a, b) {
return (a[1] < b[1] ? -1 : (a[1] > b[1] ? 1 : 0));
});
// [["bar",15],["me",75],["you",100],["foo",116]]
var orderedList = {};
for (var idx in sortable) {
orderedList[sortable[idx][0]] = sortable[idx][1];
}
return orderedList;
}
sortAssocObject(list);
// {bar: 15, me: 75, you: 100, foo: 116}
Object.entries
的答案,它是自ES2017以来最干净、最易读的现代方法:https://dev59.com/LXNA5IYBdhLWcg3wKacx#37607084 - jakub.g