개발 공부/웹개발

자바스크립트_기본 함수 구현하기(_.filter, _.uniq, _.each,_.map, _.reduce)

크롱이크 2021. 5. 19. 15:39

이번 주에는 자바스크립트 유틸리티 라이브러리 underscore의 함수들을 직접 구현하는 과제를 받았습니다. 

 

그중에서 제가 중요하다고 생각하는 몇가지만 정리해보겠습니다.

_.each

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
 *  배열 arr을 입력받을 경우, iteratee(ele, idx, arr)
 *  객체 obj를 입력받을 경우, iteratee(val, key, obj)
 * 이처럼 collection의 모든 정보가 iteratee의 인자로 잘 전달되어야 모든 경우를 다룰 수 있습니다.
 * 실제로 전달되는 callback 함수는 collection의 모든 정보가 필요하지 않을 수도 있습니다.
 */
 
// _.each는 명시적으로 어떤 값을 리턴하지 않습니다. Array.prototype.forEach
_.each = function (collection, iteratee) {
  // TODO: 여기에 코드를 작성합니다.
  //? 배열일 경우
  if(Array.isArray(collection)){
    for(let i=0; i < collection.length ; i++ ){
      iteratee(collection[i] , i , collection);
    }
  }
//? 객체일 경우
  else if(typeof collection === 'object' && collection !==null){
    for(let key in collection){
      iteratee(collection[key] , key , collection )
    }
  }
};
forEach는 라는 함수를 구현해보았다. 배열일때와 객체일때 전달되어져야 하는 인자가 다르다. 
배열인 경우에는 (요소,  인덱스, 배열)이 주어지고,
객체는 (키값, 키, 객체) 가 주어져야한다.

_.indexOf

1
2
3
4
5
6
7
8
9
10
11
12
_.indexOf = function (arr, target) {
 
  let result = -1;
 
  _.each(arr, function (item, index) {
    if (item === target && result === -1) {
      result = index;
    }
  });
 
  return result;
};
cs
indexOf는 배열안에 찾는 요소가 있을시 인덱스 넘버를 리턴해주는 함수이다.
아무것도 없을때는 -1을 반환한다.
target과 위에서 만든 _.each함수에서 item이 같다면, 인덱스 넘버를 리턴한다.

_.filter

1
2
3
4
5
6
7
8
9
10
11
12
13
// _.filter는 test 함수를 통과하는 모든 요소를 담은 새로운 배열을 리턴합니다.
 
_.filter = function (arr, test) {
  
  let result = [];
  _.each(arr, function(el){
    if(test(el)){
      result.push(el);
    }
  })
  return result;
};
 
cs
_.filter 함수는 배열안의 요소를 순회하며 조건에 맞는 요소를 골라 새로운 배열에 추가하여 리턴하는 함수이다.
_.each를 활용하여 요소마다 접근하여 진행하게된다.

_.uniq

1
2
3
4
5
6
7
8
9
10
11
12
// _.uniq는 주어진 배열의 요소가 중복되지 않도록 새로운 배열을 리턴합니다.
 
_.uniq = function (arr) {
  let result = [];
 
  _.each(arr, function(el){
    if(_.indexOf(result, el) === -1){
      result.push(el);
    }
  });
  return result;
};
cs
배열안에 중복되는 요소를 제거하여 새로운 배열로 리턴하는 함수이다.
여기서는 위에서 만든 indexOf를 사용하였다. 존재하지 않다면 -1이 나오기에 -1이 나온다면 새로운 배열에 요소를 push한다.

_.map

1
2
3
4
5
6
7
8
9
10
11
// _.map은 iteratee(반복되는 작업)를 배열의 각 요소에 적용(apply)한 결과를 담은 새로운 배열을 리턴한다.
// 함수의 이름에서 드러나듯이 _.map은 배열의 각 요소를 다른 것(iteratee의 결과)으로 매핑(mapping)한다.
 
_.map = function (arr, iteratee) {
 
  let result=[];
  _.each(arr, function(el){
    result.push(iteratee(el));
  });
  return result;
};
cs
map이라는 함수는 함수를 적용시켜 새로운 배열로 만든다.
filter와는 다르게 조건이 들어가서 일치하는 것만 꺼내는 것이 아니라, 전체에 함수를 적용시켜 새로운 배열에 push 해야한다.

_.reduce

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
//_.reduce는
//  1. 배열을 순회하며 각 요소에 iteratee 함수를 적용하고,
//  2. 그 결과값을 계속해서 누적(accumulate)합니다.
//  3. 최종적으로 누적된 결과값을 리턴합니다.
 
_.reduce = function (arr, iteratee, initVal) {
  let acc = initVal;
 
  _.each(arr, function (item, idx, src) {
    if (initVal === undefined && idx === 0) {
     acc = item;
    } else {
     acc = iteratee(acc, item, idx, src);
    }
  });
 
  return accumulator;
};
cs
가장 까다로웠다. reduce도 3개의 인자를 받아온다. (배열, 함수, 초기값) 
acc라는 변수에 초기값(initVal)이 있다면 넣어줘야한다. 
만약 undefined거나 0이라면, 배열의 첫 요소가 acc가 된다.
 each는 배열안의 요소 하나씩 접근하는 요소이므로 다시 돌며 else로 들어가 계속해서 함수를 적용하여, acc에 재할당해준다.
누적된 결과값을 알 수있다. 

 

자바스크립트에서 활용하는 함수를 구현해 보았습니다.

함수들을 구현해봄으써 그 함수가 어떻게 활용되고 있는지 자세히 알게 되었습니다. 이 과제를 통해 코드스테이츠에서 왜 이런 과제를 냈는지 이해가 되는 시간이였습니다.  

반응형