按属性值排序JavaScript对象

Ste*_*ike 629 javascript sorting properties object

如果我有一个JavaScript对象,例如:

var list = {
  "you": 100, 
  "me": 75, 
  "foo": 116, 
  "bar": 15
};
Run Code Online (Sandbox Code Playgroud)

有没有办法根据值对属性进行排序?所以我最终得到了

list = {
  "bar": 15, 
  "me": 75, 
  "you": 100, 
  "foo": 116
};
Run Code Online (Sandbox Code Playgroud)

Nos*_*dna 646

将它们移动到一个数组,对该数组进行排序,然后将该数组用于您的目的.这是一个解决方案:

var maxSpeed = {
    car: 300, 
    bike: 60, 
    motorbike: 200, 
    airplane: 1000,
    helicopter: 400, 
    rocket: 8 * 60 * 60
};
var 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]]
Run Code Online (Sandbox Code Playgroud)

一旦拥有了数组,就可以按照自己喜欢的顺序从数组中重建对象,从而实现您的目标.这可以在我所知道的所有浏览器中运行,但它将依赖于实现的怪癖,并且可能随时中断.您永远不应该假设JavaScript对象中元素的顺序.

  • 这是一个更紧凑的代码版本.Object.keys(maxSpeed).sort(function(a,b){return - (maxSpeed [a] - maxSpeed [b])}); (30认同)
  • `_.pairs`将对象变为[[key1,value1],[key2,value2]].然后调用sort.然后在它上面调用`_.object`将其转回. (30认同)
  • 你能否重新表述"你可以重建"到"你可以使用数组来维护键的排序和从对象中提取值"吗?正如你自己提到的,这不仅是非标准的,这个错误的假设被更多的浏览器打破,而不仅仅是今天的Chrome,因此最好不要鼓励用户尝试. (23认同)
  • @Funkodebat AFAIK这个问题与下划线库无关. (10认同)
  • 我疯了还是ES10解决方案不起作用?当我尝试时,Object.fromEntries 创建一个按键排序的对象,撤消所有排序工作。 (7认同)
  • @TheBrain:只是要添加,`keys()`只有IE9 +(和其他现代浏览器)支持,如果这是关注的话.另外`keys()`从元素原型链中排除了可枚举的属性(与for..in不同) - 但这通常是更理想的. (5认同)
  • 我的项目需要对象键来干净地合并,但也需要显式排序,因为元数据驱动 UI。我遵循了类似的方法,只是我添加了一个 hasOwnProperty 检查以避免爬上原型链。这是一篇关于在 JS 中迭代对象属性的好文章 https://hackernoon.com/5-techniques-to-iterate-over-javascript-object-entries-and-their-performance-6602dcb708a8 (2认同)
  • ES8 示例不起作用,并且不知道如何修复。 (2认同)
  • 同样这里 ES10 解决方案不起作用 (2认同)
  • 对于那些想知道这些 lambda 参数是什么的人。`sort(([,a],[,b]) => ab)` 这意味着只需为条目的值声明一个变量。使用此语法 [key, value] 进行提交。 (2认同)

Mar*_*s R 377

我们不想复制整个数据结构,也不想使用需要关联数组的数组.

这是与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
Run Code Online (Sandbox Code Playgroud)

  • 它按值排序,并显示键 - 但在打印时我们会丢失值计数,因为它只打印键. (24认同)
  • 这似乎是按键而不是值进行排序,这不是问题所要求的? (21认同)
  • 如果你将`.map(key => list [key]);`添加到排序的末尾,它将返回整个对象而不仅仅是键 (14认同)
  • JavaScript中不保证对象属性顺序,因此应该对数组进行排序,而不是对象(这就是你所说的'关联数组'). (6认同)
  • 别忘了.`keysSorted`是一个数组!不是对象! (6认同)
  • @JamesMoran 添加只会返回该键的值,而不是整个对象。 (5认同)
  • @ChristopherMeyers排序*是在数组中完成的.它是从Object.keys()获得的,然后通过比较原始对象中的值进行排序. (4认同)
  • 这确实有效(最简单).谢谢! (3认同)
  • @MarkusR就是这样.就是这样.不确定我的问题是什么.好答案! (3认同)
  • 同意@DollarAkshay,相反,需要使用 `.map((key) => ({ [key]: list[key] }))` 来返回对象数组。 (2认同)

ino*_*nik 169

您的对象可以包含任意数量的属性,如果将对象放在数组中,则可以选择按所需的任何对象属性,数字或字符串进行排序.考虑这个数组:

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'
    }
];
Run Code Online (Sandbox Code Playgroud)

按出生日期排序,最早出现

// 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);
Run Code Online (Sandbox Code Playgroud)

按名称分类

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);
Run Code Online (Sandbox Code Playgroud)

http://jsfiddle.net/xsM5s/16/

  • @JemarJones`localCompare`看起来像一个非常有用的功能!请注意,每个浏览器都不支持它 - IE 10及更低版本,Safari Mobile 9及更低版本. (2认同)

Sta*_*ano 61

为了完整起见,此函数返回已排序的对象属性数组:

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) { 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}]
Run Code Online (Sandbox Code Playgroud)

以上代码的Jsfiddle就在这里.此解决方案基于本文.

这里有用于排序字符串的更新小提琴.您可以从中删除其他.toLowerCase()转换,以进行区分大小写的字符串比较.


Nic*_*itz 36

JavaScript对象按定义排序(请参阅ECMAScript语言规范,第8.6节).语言规范甚至不保证,如果你连续两次迭代一个对象的属性,它们将在第二次以相同的顺序出现.

如果需要订购内容,请使用数组和Array.prototype.sort方法.

  • EcmaScript 5实际上将枚举的顺序定义为插入顺序 - 缺少定义被认为是ES3中的规范错误.值得注意的是,EcmaScript规范定义了没有人会认为理智的行为 - 例如规范行为是在运行时抛出语法错误,不正确地使用continue,break,++, - ,const等等.到达规范任何在到达该代码之前抛出异常的引擎都是错误的* (8认同)
  • 该错误报告是不合理的,那些依赖于未记录行为的人是那些有错误的人.阅读ECMASCript第8.6节; 它明确指出"一个对象是一个无序的属性集合".任何人发现在一些实现中似乎没有这种方式,然后开始依赖于特定于实现的行为,犯了一个大错误,他们不应该试图将责任推离自己.如果我在Chrome团队中,我会将该错误报告标记为"Invalid,WontFix". (6认同)
  • 请注意,对此有很多争论.大多数实现使列表按照添加元素的顺序排列.IIRC,Chrome没有.有人争论Chrome是否应该与其他实现一致.我认为JavaScript对象是一个哈希值,不应该假设它.我相信Python经历了相同的论点,并且最近引入了一个新的有序类似哈希的列表.对于大多数浏览器,您可以通过重新创建对象,按排序值添加元素来执行您想要的操作.但你不应该. (4认同)
  • @olliej - 我不认为这是正确的.规范说:"枚举属性的机制和顺序(第一个算法中的步骤6.a,第二个算法中的步骤7.a)未指定".这是最终草案PDF的第92页. (4认同)

Jas*_*han 35

@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
Run Code Online (Sandbox Code Playgroud)

更新:20174月 - 返回myObj上面定义的排序对象.

Object
 .keys(myObj)
 .sort((a, b) => myObj[a]-myObj[b])
 .reduce((_sortedObj, key) => ({
   ..._sortedObj, 
   [key]: myObj[key]
 }), {})
Run Code Online (Sandbox Code Playgroud)

试试吧!

更新:201810月 - Object.entries版本

Object
 .entries(myObj)
 .sort()
 .reduce((_sortedObj, [k,v]) => ({
   ..._sortedObj, 
   [k]: v
 }), {})
Run Code Online (Sandbox Code Playgroud)

试试吧!

  • OP的问题,以及这个答案不包含嵌套对象@Rohanil也许你想问另一个问题,而不是向下投票.您的各种类型的嵌套对象显然需要比此解决方案提供的更多 (4认同)
  • 不适用于 `var myObj = {"1": {"Value": 40}, "2": {"Value": 10}, "3": {"Value": 30}, "4": { “值”:20}};` (2认同)

kid*_*ing 29

ECMAScript 2017介绍Object.values / Object.entries.顾名思义,前者将对象的所有值聚合到一个数组中,后者将整个对象聚合成一个数组[key, value]数组; 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]]
Run Code Online (Sandbox Code Playgroud)

  • 为了进一步简化,请将过程合并为一行: ```letsorted = Object.entries(obj).sort((a, b) =&gt; a[1] - b[1]); ```` (6认同)
  • @vsync此答案给出的结果与接受的答案相同,但是代码更少且没有临时变量。 (2认同)
  • FWIW,这个答案很干净,是唯一对我有帮助的答案。 (2认同)

Ali*_*eza 22

好吧,你可能知道,javascript有sort()函数,对数组进行排序,但没有任何对象...

所以在这种情况下,我们需要以某种方式获取键的数组并对它们进行排序,这就是api在大多数情况下为数组提供对象的原因,因为Array具有比对象文字更多的本机函数,无论如何,快速解决方法是使用Object.key返回一个对象键数组,我创建了ES6函数,在其下面为你完成了这项工作,它在javascript中使用了native sort()reduce()函数:

function sortObject(obj) {
  return Object.keys(obj)
    .sort().reduce((a, v) => {
    a[v] = obj[v];
    return a; }, {});
}
Run Code Online (Sandbox Code Playgroud)

现在你可以像这样使用它:

let myObject = {a: 1, c: 3, e: 5, b: 2, d: 4};
let sortedMyObject = sortObject(myObject);
Run Code Online (Sandbox Code Playgroud)

检查sortedMyObject,您可以看到按键排序的结果,如下所示:

{a: 1, b: 2, c: 3, d: 4, e: 5}
Run Code Online (Sandbox Code Playgroud)

这样,主要对象将不会被触及,我们实际上获得了一个新对象.

我还创建了下面的图像,以使功能步骤更加清晰,以防您需要更改它以便按照您的方式工作:

按属性值排序javascript对象

  • 这是按键而不是按值排序. (4认同)
  • @VtoCorleone 这只是一个自然的排序错误,1 排在第一位,而不是所提出的算法中的缺陷。 (2认同)

Vas*_*lov 9

var 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}
Run Code Online (Sandbox Code Playgroud)


小智 8

在没有多个 for 循环的情况下对值进行排序(按键排序,将排序回调中的索引更改为“0”)

const list = {
    "you": 100, 
    "me": 75, 
    "foo": 116, 
    "bar": 15
  };

let sorted = Object.fromEntries(
                Object.entries(list).sort( (a,b) => a[1] - b[1] )    
             ) 
console.log('Sorted object: ', sorted) 
Run Code Online (Sandbox Code Playgroud)


Mar*_*aei 8

很短很简单!

var sortedList = {};
Object.keys(list).sort((a,b) => list[a]-list[b]).forEach((key) => {
    sortedList[key] = list[key]; });
Run Code Online (Sandbox Code Playgroud)


jul*_*ljk 7

使用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
Run Code Online (Sandbox Code Playgroud)


Rom*_*din 6

Underscore.js或Lodash.js用于高级数组或对象排序

 var data={
        "models": {

            "LTI": [
                "TX"
            ],
            "Carado": [
                "A",
                "T",
                "A(????)",
                "A(????)",
                "T(????)",
                "T(????)",
                "A",
                "T"
            ],
            "SPARK": [
                "SP110C 2",
                "sp150r 18"
            ],
            "Autobianchi": [
                "A112"
            ]
        }
    };

    var arr=[],
        obj={};
    for(var i in data.models){
      arr.push([i, _.sortBy(data.models[i],function (el){return el;})]);
    }
    arr=_.sortBy(arr,function (el){
      return el[0];
    });
    _.map(arr,function (el){return obj[el[0]]=el[1];});
     console.log(obj);
Run Code Online (Sandbox Code Playgroud)

演示


bon*_*nna 5

我正在遵循slebetman给出的解决方案(请仔细阅读所有细节),但调整后,因为您的对象是非嵌套的.

// First create the array of keys/values so that we can sort it:
var sort_array = [];
for (var key in list) {
    sort_array.push({key:key,value:list[key]});
}

// Now sort it:
sort_array.sort(function(x,y){return x.value - y.value});

// Now process that object with it:
for (var i=0;i<sort_array.length;i++) {
    var item = list[sort_array[i].key];

    // now do stuff with each item
}
Run Code Online (Sandbox Code Playgroud)