Cum pentru a rezuma valorile echivalente cheile într-o matrice de obiecte în cazul în care nu poate fi mai multe, nou generat cheile?

0

Problema

Există o serie de obiecte ca atât în cazul în care există o "categorie" cheie și unele "serie" cheile.

arrOne = [
    {
        "series_1": 25,
        "category": "Category 1",
        "series_2": 50
    },
    {
        "series_1": 11,
        "category": "Category 2",
        "series_2": 22
    },
    {
        "series_1": 32,
        "category": "Category 1",
        "series_2": 74
    },
    {
        "series_1": 74,
        "category": "Category 3",
        "series_2": 98
    },
    {
        "series_1": 46,
        "category": "Category 3",
        "series_2": 29
    },

]

(Rețineți că o "categorie" poate fi destul de mult orice valoare, deși nu va fi probabil mai multe valori similare, precum și unele valori unice exemplu, există mai multe obiecte cu categoria de "valoare" Categoria 3', dar numai 1 cu categoria de "valoare" Categoria 2')

Următoarele linii de cod se va adăuga până toate series_1 pentru obiecte cu aceeași categorie

        var objForAllCategories = {};
        this.arrOne.forEach(item => {
            if (objForAllCategories.hasOwnProperty(item.category))
                objForAllCategories[item.category] = objForAllCategories[item.category] + item.series_1;
            else
                objForAllCategories[item.category] = item.series_1;
        });
        for (var prop in objForAllCategories) {
            this.allCategoriesAndValues.push({ 
                category: prop, 
                series_1: objForAllCategories[prop] 
            });
        }

Deci ar rezulta:

allCategoriesAndValues = [
    {
        "category": "Category 1",
        "series_1": 57       // 25 + 32 adding up series_1 from all 'Category 1' items in arrOne
    },
    {
        "category": "Category 2",
        "series_1": 11      // only 1 'Category 2' from arrOne
    },
    {
        "category": "Category 3",
        "series_1": 120     // 74 + 46 adding up series_1 from all 'Category 3' items in arrOne
    }
]

Cu toate acestea, vreau să fie în măsură pentru a adăuga nu doar series_1 dar, de asemenea, toate celelalte elemente.

Acest exemplu are doar categoria și series_1 și series_2 ca cheile. Cu toate acestea, ar putea fi:

  1. series_3
  2. series_4
  3. series_5
  4. series_6
  5. series_7
  6. etc..

Cum pot cont pentru toți potențialii series_x?

Destinat rezultat:

allCategoriesAndValues = [
    {
        "category": "Category 1",
        "series_1": 57,
        "series_2": 124,
        ..... if 'series_3', 'series_4' etc. existed, it would be included in this as above
    },
    {
        "category": "Category 2",
        "series_1": 11,
        "series_2": 22,
        ..... if 'series_3', 'series_4' etc. existed, it would be included in this as above
    },
    {
        "category": "Category 3",
        "series_1": 120,
        "series_2": 127,
        ..... if 'series_3', 'series_4' etc. existed, it would be included in this as above
    }
]
arrays javascript json key-value
2021-11-24 02:19:06
6

Cel mai bun răspuns

2

Să se ocupe de mai multe proprietăți logice, puteți bucla prin fiecare proprietate și pentru a verifica dacă se potrivește cu regex series_\d+. Dacă nu, știi că nu este o proprietate care aveți nevoie pentru a crește, și-l ocupe în mod corespunzător (o proprietate existenta verifica, de asemenea, este necesar, după cum a subliniat de către Jayce444).

Următoarea soluție foloseste Array.reduce. În reductor funcție, se verifică dacă acumulatorul matrice conține un element cu același category proprietatea ca în prezent în curs de buclă prin. Dacă nu, se va incrementa de proprietăți corespunzătoare. În caz contrar, se va împinge elementul curent a acumulatorului matrice.

arrOne=[{series_1:25,category:"Category 1",series_2:50},{series_1:11,category:"Category 2",series_2:22},{series_1:32,category:"Category 1",series_2:74},{series_1:74,category:"Category 3",series_2:98},{series_1:46,category:"Category 3",series_2:29,series_3:50}];

const res = arrOne.reduce((a, b) => {
  let found = a.find(e => e.category == b.category)
  if (found) {
    Object.keys(b).forEach(e => {
      if (/series_\d+/g.test(e)) found[e] = found[e] ? found[e] + b[e] : b[e];
    })
  } else {
    a.push(b)
  }
  return a;
}, [])

console.log(res)

2021-11-24 02:38:52

Această abordare se rupe atunci când toate obiectele nu au aceeași serie de chei. De exemplu, dacă adăugați series_3: 5 la primul obiect numai, se sfârșește ca series_3: NaN în rezultat.
Jayce444

Nu ratați noua serie cheile adăugat pentru aceeași categorie mai târziu obiecte? Este, de asemenea, O(n^2) (cred)
Phil

@Phil Multumesc pentru notificare. Am actualizat răspunsul meu.
Spectric
1

Ceva de genul asta ar putea funcționa.

arrOne = [ { "series_1": 25, "category": "Category 1", "series_2": 50 }, { "series_1": 11, "category": "Category 2", "series_2": 22 }, { "series_1": 32, "category": "Category 1", "series_2": 74 }, { "series_1": 74, "category": "Category 3", "series_2": 98 }, { "series_1": 46, "category": "Category 3", "series_2": 29 },];

const result = [];
arrOne.reduce((acc, {category, ...series}) => {
  if (acc.has(category)) {
    Object.entries(series).forEach(([key, value]) => {
      if (key.startsWith('series_')) {
        acc.get(category)[key] = (acc.get(category)[key] || 0) + value;
      }
    });
  } else {
    const item = {category, ...series};
    result.push(item);
    acc.set(category, item);
  }
  return acc;
}, new Map());

console.log(result);

2021-11-24 02:27:40

Nu sunt sigur despre un reductor cu efecte secundare (result mutație)
Phil
0

Creați o Hartă pentru a colaționa serie de sume de categorie.

Apoi a crea o serie de harta cu tastele ca category

const arr1 = [{"series_1":25,"category":"Category 1","series_2":50},{"series_1":11,"category":"Category 2","series_2":22},{"series_1":32,"category":"Category 1","series_2":74},{"series_1":74,"category":"Category 3","series_2":98},{"series_1":46,"category":"Category 3","series_2":29}]

const t1 = performance.now()

const cats = arr1.reduce((map, { category, ...series }) =>
  map.set(category, Object.entries(series)
    .reduce((s, [ key, count ]) => ({
      ...s,
      [ key ]: (s[key] ?? 0) + count
    }), map.get(category) ?? {})
  ), new Map())

const allCategoriesAndValues = Array.from(cats, ([ category, series ]) => ({
  category,
  ...series
}))

const t2 = performance.now()

console.info(allCategoriesAndValues)
console.log(`Took ${t2 - t1}ms`)
.as-console-wrapper { max-height: 100% !important; }

2021-11-24 02:32:20
0

Voi face asta în acest fel...

const arrOne = 
  [ { series_1: 25, category: 'Category 1', series_2: 50 } 
  , { series_1: 11, category: 'Category 2', series_2: 22 } 
  , { series_1: 32, category: 'Category 1', series_2: 74 } 
  , { series_1: 74, category: 'Category 3', series_2: 98 } 
  , { series_1: 46, category: 'Category 3', series_2: 29 } 
  ] 

console.time('chrono')

const allCategoriesAndValues =
  Object.entries(
  arrOne.reduce((r,{ category, ...series })=>
    {
    let cat = r[category] = r[category] ?? {} 
    Object.entries(series).forEach(([sName,val]) => cat[sName] = (cat[sName] ?? 0) + val);
    return r
    },{})
  ).map(([category,series])=>({category,...series}))

console.timeEnd('chrono')

console.log( allCategoriesAndValues )
.as-console-wrapper {max-height: 100%!important;top:0 }

2021-11-24 02:47:52
0

Ai putea repeta peste largă de obiecte și apoi cheile de fiecare obiect, stocarea într-un tampon de obiect. Trebuie doar să fie verificați pentru existența de fiecare cheie și adăugați-l în cazul în care lipsește, sau puteți doar să coaguleze falsey cheile într-o valoare implicită, cum am făcut eu. Am elimina categorie cheie de obiect după ce am obține valoarea sa, astfel încât să nu trebuie să încercați să-l sări în cheie iterație.

const arrOne = [
  {"series_1": 25, "category": "Category 1", "series_2": 50},
  {"series_1": 11, "category": "Category 2", "series_2": 22},
  {"series_1": 32, "category": "Category 1", "series_2": 74},
  {"series_1": 74, "category": "Category 3", "series_2": 98},
  {"series_1": 46, "category": "Category 3", "series_2": 29},
];

let buffer = {};
arrOne.forEach(i=>{
  let c = i.category;
  buffer[c] = buffer[c] || {};
  delete i.category;
  Object.keys(i).forEach(k=>{
    buffer[c][k] = buffer[c][k] || 0;
    buffer[c][k] += i[k];
  });
});

console.log(buffer);

let final = Object.keys(buffer).map(k=>{return {[k]: buffer[k]}});
console.log(final);

Dacă nu aveți nevoie de acest lucru într-o matrice, ultimul pas este opțional. Ea există doar pentru a transforma obiectul intr-o matrice.

2021-11-24 02:31:18
0

Iată cum aș face-o

const res = arrOne.reduce((acc, { category, ...vals }) => {
    if (acc[category]) {
        Object.entries(vals).forEach(([ key, val ]) => acc[category][key] = acc[category][key] ? acc[category][key] + val : val);

    } else {
        acc[category] = vals;

    }

    return acc;
}, {});
2021-11-24 03:11:13

În alte limbi

Această pagină este în alte limbi

Русский
..................................................................................................................
Italiano
..................................................................................................................
Polski
..................................................................................................................
한국어
..................................................................................................................
हिन्दी
..................................................................................................................
Français
..................................................................................................................
Türk
..................................................................................................................
Česk
..................................................................................................................
Português
..................................................................................................................
ไทย
..................................................................................................................
中文
..................................................................................................................
Español
..................................................................................................................
Slovenský
..................................................................................................................