Улучшить базу деления массива на его элемент - PullRequest
0 голосов
/ 30 мая 2018

Я хочу разделить массив на два массива на основе его элементов, например, если у меня есть этот массив в качестве входных данных

[{id:1,name:"item1_g1"},{id:2,name:"item2_g2"},{id:3,name:"item3_g1"},{id:2,name:"item4_g2"}]

, выходные данные должны быть двумя массивами, подобными этому

первый массив:

[{id:1,name:"item1_g1"},{id:3,name:"item3_g1"}]

второй массив:

[{id:2,name:"item2_g2"},{id:2,name:"item4_g2"}]

Поскольку вам нужно просто сгруппировать элементы на основе их атрибута name

Я использую angularjs и я ужепопробовал этот код

var values = [{
  id: 1,
  name: "item1_g1"
}, {
  id: 2,
  name: "item2_g2"
}, {
  id: 3,
  name: "item3_g1"
}, {
  id: 2,
  name: "item4_g2"
}];
var group1 = values.filter(function(item) {
  return (item.name.includes('g1'));
});
var group2 = values.filter(function(item) {
  return (item.name.includes('g2'));
});
console.log(group1, group2)

Но мне не понравилось это решение, потому что в случае, если у меня большой массив, мне нужно его зациклить n раз, чтобы извлечь n массивов, что плохо для производительности

Есть ли способ сделать все это за один цикл?

Ответы [ 4 ]

0 голосов
/ 30 мая 2018
// I would do this the old fashioned way. 
// iterate over values and add them to a map with same group id
// this way you won't need to worry about how many groups you have


var groups = {};
for(i=0;i<values.length;i++){
    var groupNum = values[i].name.substring(values[i].name.indexOf('_g') + 2);
    if(groups[groupNum]){
        groups[groupNum].push(values[i]);
    }
    else {
        groups[groupNum] = [values[i]];
    }
}

console.log(groups);
{
  "1": [
    {
      "id": 1,
      "name": "item1_g1"
    },
    {
      "id": 3,
      "name": "item3_g1"
    }
  ],
  "2": [
    {
      "id": 2,
      "name": "item2_g2"
    },
    {
      "id": 2,
      "name": "item4_g2"
    }
  ]
}"
0 голосов
/ 30 мая 2018

Вы можете выполнить цикл только один раз, используя Array.prototype.reduce () в сочетании с String.prototype.slice ()

Пример кода:

const values = [{id: 1,name: "item1_g1"}, {id: 2,name: "item2_g2"}, {id: 3,name: "item3_g1"}, {id: 2,name: "item4_g2"}];
const groups = values.reduce((a, c) => (a[c.name.slice(-2)].push(c), a), {g1: [], g2: []});

console.log(groups);
0 голосов
/ 30 мая 2018

Вы можете сделать это за один цикл, как показано ниже:

var values = [{
  id: 1,
  name: "item1_g1"
}, {
  id: 2,
  name: "item2_g2"
}, {
  id: 3,
  name: "item3_g1"
}, {
  id: 2,
  name: "item4_g2"
}];
let group1 = [];
let group2 = [];
values.forEach( item => {
    if(item.name.includes('g1')){
       group1.push(item);
    } else {
       group2.push(item);
    }
})
console.log(group1)
console.log(group2)
0 голосов
/ 30 мая 2018

Просмотрите массив всего один раз, используйте критерии, чтобы решить, в какой массив должен быть помещен элемент.Например:

var values = [
  { id: 1, name: "item1_g1" }, 
  { id: 2, name: "item2_g2" }, 
  { id: 3, name: "item3_g1" }, 
  { id: 2, name: "item4_g2" }
];

const [group1, group2] = values.reduce((groups, el) => {
   groups[1^el.name.includes('g1')].push(el);  
   return groups;
}, [[], []]);
  
console.log(group1, group2);

Вы можете создать более общую функцию группировки, которая принимает два аргумента - массив элементов для сортировки и функцию предиката (которая возвращает индекс).Это будет немного более многословно, хотя:

function groupBy(arr, indexer) {
  return arr.reduce((groups, el) => {
    const index = indexer(el);
    groups[index] = groups[index] || [];
    groups[index].push(el);
    return groups;
  }, []);
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...