admin管理员组文章数量:1339481
I have an array of objects, let's say:
var objects = [
{name: 'A'},
{name: '1'},
{name: 'B'}
]
Knowing that I can sort it using Lodash sortBy:
objects= _.sortBy(objects, 'name')
which will result in this:
[
{name: '1'},
{name: 'A'},
{name: 'B'}
]
But my desired output is this:
[
{name: 'A'},
{name: 'B'},
{name: '1'}
]
Please help.
I have an array of objects, let's say:
var objects = [
{name: 'A'},
{name: '1'},
{name: 'B'}
]
Knowing that I can sort it using Lodash sortBy:
objects= _.sortBy(objects, 'name')
which will result in this:
[
{name: '1'},
{name: 'A'},
{name: 'B'}
]
But my desired output is this:
[
{name: 'A'},
{name: 'B'},
{name: '1'}
]
Please help.
Share Improve this question edited Dec 27, 2016 at 6:35 user663031 asked Dec 27, 2016 at 2:55 Zhi RuiZhi Rui 1,0631 gold badge11 silver badges10 bronze badges 9- @TrojanByAccident Yes what do you mean by same key? They are three different objects. And I'd showed what I have tried using Lodash soryBy, what do you mean by 'put some effort into it' when I posted what I have tried? Flagging your ment. – Zhi Rui Commented Dec 27, 2016 at 3:00
- What's the possible range of characters? Only A-Z and 0-9? – Robby Cornelissen Commented Dec 27, 2016 at 3:01
- @RobbyCornelissen Yes sir. – Zhi Rui Commented Dec 27, 2016 at 3:02
- 1 @user1422866 That's not exactly showing what you have tried. That's saying that you know something else doesn't give you the desired results, but you aren't showing any code of your own that attempts to achieve that result. – TrojanByAccident Commented Dec 27, 2016 at 3:02
-
1
When sorting the values
AA
andA1
, which do you expect to e first? I.e., do your sorting rules only apply to the first character or also to subsequent characters? – Robby Cornelissen Commented Dec 27, 2016 at 4:35
5 Answers
Reset to default 6Using Array#sort
you can apply this logic:
// If both are numbers or both are not numbers
isNaN(a.name) === isNaN(b.name) ?
// then pare between them
a.name.localeCompare(b.name)
: // else
// If the 1st is not a number move it up, if it's a number move it down
(isNaN(a.name) ? -1 : 1);
Without lodash:
var objects = [{"name":"A"},{"name":"3"},{"name":"1"},{"name":"B"}];
objects.sort(function(a, b) {
return isNaN(a.name) === isNaN(b.name) ? a.name.localeCompare(b.name) : (isNaN(a.name) ? -1 : 1);
});
console.log(objects);
As part of a lodash's chain:
var objects = [{"name":"A"},{"name":"3"},{"name":"1"},{"name":"B"}];
var result = _(objects)
.sort(function(a, b) {
return isNaN(a.name) === isNaN(b.name) ? a.name.localeCompare(b.name) : (isNaN(a.name) ? -1 : 1);
})
.value();
console.log(result);
<script src="https://cdnjs.cloudflare./ajax/libs/lodash.js/4.17.3/lodash.min.js"></script>
I'm not sure if using lodash's sortBy
is the correct approach for this problem. Here's an implementation using Javascript's Array#sort
method.
It takes not only the first character but the entire string into account when doing the sorting.
const objects = [{
name: '2'
}, {
name: 'B'
}, {
name: '1'
}, {
name: 'A'
}, {
name: 'A1'
}, {
name: 'AA'
}]
objects.sort((o1, o2) => {
let a = o1.name, b = o2.name;
let isDigit = x => x >= 48 && x <= 57;
for (let i = 0, n = Math.min(a.length, b.length); i < n; i++) {
let aCharCode = a.charCodeAt(i), bCharCode = b.charCodeAt(i);
if (aCharCode !== bCharCode) {
if (isDigit(aCharCode) && !isDigit(bCharCode)) return 1;
if (isDigit(bCharCode) && !isDigit(aCharCode)) return -1;
return aCharCode - bCharCode;
}
}
return a.length - b.length;
});
console.log(objects)
For the given input, this prints out
[
{
"name": "A"
},
{
"name": "AA"
},
{
"name": "A1"
},
{
"name": "B"
},
{
"name": "1"
},
{
"name": "2"
}
]
Solution with two _.sortBy()
One for priotizing alphabets first, then another one for sorting the elements.
In my opinion it's more readable and it will have no performance impact.
const objects = [
{name: 'B'},
{name: '2'},
{name: '1'},
{name: 'A'}
]
const result = _.sortBy(_.sortBy(objects, o => !isNaN(parseInt(o.name)), 'name'))
console.log(result)
<script src="https://lodash./vendor/cdn.jsdelivr/lodash/4.17.3/lodash.min.js"></script>
Partition the input, sort separately and join back:
Code:
const _ = require('lodash');
const customSort = (a) => _.chain(_.partition(a, i => isNaN(i.name))).flatMap(p => _.sortBy(p, 'name')).value();
const input = [
{ name: '1' },
{ name: 'A' },
{ name: '6' },
{ name: 'B' },
{ name: 'a' },
{ name: '0' },
{ name: '3' }];
console.log(customSort(input));
Output:
[ { name: 'A' },
{ name: 'B' },
{ name: 'a' },
{ name: '0' },
{ name: '1' },
{ name: '3' },
{ name: '6' } ]
If name
is an integer prefix it with z
for parison.
var objects = [
{name: 'z'},
{name: 'A'},
{name: '1'},
{name: 'B'}
], sorted = _.sortBy( objects, [
function(d) {
return !isNaN(parseFloat(d.name))
&& isFinite(d.name)
? 'z' + d.name
: d.name;
}
]
);
console.log(sorted);
<script src="https://lodash./vendor/cdn.jsdelivr/lodash/4.17.3/lodash.min.js"></script>
版权声明:本文标题:javascript - Lodash: Sort array of objects prioritizing alphabets first followed by numbers - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743582105a2505881.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论