[javascript] lodash .groupBy 사용. 그룹화 된 출력에 자신의 키를 추가하는 방법은 무엇입니까?

이 샘플 데이터가 API에서 반환되었습니다.

Lodash를 사용 _.groupBy하여 데이터를 더 잘 사용할 수있는 개체로 변환하고 있습니다. 반환 된 원시 데이터는 다음과 같습니다.

[
    {
        "name": "jim",
        "color": "blue",
        "age": "22"
    },
    {
        "name": "Sam",
        "color": "blue",
        "age": "33"
    },
    {
        "name": "eddie",
        "color": "green",
        "age": "77"
    }
]

_.groupBy함수가 다음과 같은 객체를 반환하기를 원합니다 .

[
    {
        color: "blue",
        users: [
            {
                "name": "jim",
                "color": "blue",
                "age": "22"
            },
            {
                "name": "Sam",
                "color": "blue",
                "age": "33"
            }
        ]
    },
    {
        color: "green",
        users: [
            {
                "name": "eddie",
                "color": "green",
                "age": "77"
            }
        ]
    }
]

현재 나는

_.groupBy(a, function(b) { return b.color})

이것은 이것을 반환하고 있습니다.

{blue: [{..}], green: [{...}]}

그룹화는 정확하지만 원하는 키 ( color, users) 를 추가하고 싶습니다 . 이것이 가능 _.groupBy합니까? 또는 다른 LoDash유틸리티?



답변

Lodash 4.x에서 이렇게 할 수 있습니다.

var data = [{
  "name": "jim",
  "color": "blue",
  "age": "22"
}, {
  "name": "Sam",
  "color": "blue",
  "age": "33"
}, {
  "name": "eddie",
  "color": "green",
  "age": "77"
}];

console.log(
  _.chain(data)
    // Group the elements of Array based on `color` property
    .groupBy("color")
    // `key` is group's name (color), `value` is the array of objects
    .map((value, key) => ({ color: key, users: value }))
    .value()
);
<script src="https://cdnjs.cloudflare.com/ajax/libs/lodash.js/4.17.15/lodash.min.js"></script>


원래 답변

var result = _.chain(data)
    .groupBy("color")
    .pairs()
    .map(function(currentItem) {
        return _.object(_.zip(["color", "users"], currentItem));
    })
    .value();
console.log(result);

온라인 데모

참고 : Lodash 4.0 이상에서는 .pairs기능의 이름이_.toPairs()


답변

이렇게 간단하지 않나요?

var result = _(data)
            .groupBy(x => x.color)
            .map((value, key) => ({color: key, users: value}))
            .value();


답변

또 다른 방법

_.chain(data)
    .groupBy('color')
    .map((users, color) => ({ users, color }))
    .value();


답변

가장 많이 득표 한 답변은 현재 “왜 사용하는 것이 실수인지 나쁜 관행으로 간주되는 Lodash _.chain기능을 사용 _.chain합니다.

다음은 함수 프로그래밍 관점 에서 문제에 접근하는 몇 줄입니다 .

import tap from "lodash/fp/tap";
import flow from "lodash/fp/flow";
import groupBy from "lodash/fp/groupBy";

const map = require('lodash/fp/map').convert({ 'cap': false });

const result = flow(
      groupBy('color'),
      map((users, color) => ({color, users})),
      tap(console.log)
    )(input)

input변환하려는 배열은 어디에 있습니까 ?


답변

감사합니다 @thefourtheye , 귀하의 코드가 크게 도움이되었습니다. Lodash 버전 4.5.0을 사용하여 솔루션에서 일반 함수를 만들었습니다.

function groupBy(dataToGroupOn, fieldNameToGroupOn, fieldNameForGroupName, fieldNameForChildren) {
            var result = _.chain(dataToGroupOn)
             .groupBy(fieldNameToGroupOn)
             .toPairs()
             .map(function (currentItem) {
                 return _.zipObject([fieldNameForGroupName, fieldNameForChildren], currentItem);
             })
             .value();
            return result;
        }

그것을 사용하려면 :

var result = groupBy(data, 'color', 'colorId', 'users');

다음은 업데이트 된 피들러입니다.

https://jsfiddle.net/sc2L9dby/


답변

다음은 lodash 4 및 ES6을 사용하는 업데이트 된 버전입니다.

const result = _.chain(data)
    .groupBy("color")
    .toPairs()
    .map(pair => _.zipObject(['color', 'users'], pair))
    .value();


답변

내 라이브러리 를 사용하여 몇 줄로이 작업을 수행 할 수 있는 다른 접근 방식을 제안합니다 .

var groupMe = sequence(
  groupBy(pluck('color')),
  forOwn(function(acc, k, v) {
    acc.push({colors: k, users: v});
    return acc;
  },[])
);

var result = groupMe(collection);

lodash 또는 Underscore에서는 인수 순서가 반대이기 때문에 다소 어려울 수 _.partial있으므로 많이 사용해야 합니다.