[JS]_daily coding #32

seul·2022년 7월 7일
0

Algorithm

목록 보기
30/31

[SEB FE] Section3 Daily Coding 11_getItemFromTwoSortedArrays

문제
길이가 m, n이고 오름차순으로 정렬되어 있는 자연수 배열들을 입력받아 전체 요소 중 k번째 요소를 리턴해야 합니다.

입력
인자 1 : arr1 - 자연수를 요소로 갖는 배열
인자 2 : arr2 - 자연수를 요소로 갖는 배열
인자 3 : k - number 타입의 0 이상의 정수

출력
number 타입을 리턴해야 합니다.

주의사항
두 배열의 길이의 합은 1,000,000 이하입니다.
어떤 배열 arr의 k번째 요소는 arr[k-1]을 의미합니다.

입출력 예시

let arr1 = [1, 4, 8, 10];
let arr2 = [2, 3, 5, 9];
let result = getItemFromTwoSortedArrays(arr1, arr2, 6);
console.log(result); // --> 8

arr1 = [1, 1, 2, 10];
arr2 = [3, 3];
result = getItemFromTwoSortedArrays(arr1, arr2, 4);
console.log(result); // --> 3

Advanced
단순히 처음부터 끝까지 찾아보는 방법(O(K)) 대신 다른 방법(O(logK))을 탐구해 보세요.

힌트
이진 탐색(binary search)을 응용하여 해결합니다.


첫번째 접근

const getItemFromTwoSortedArrays = function (arr1, arr2, k) {
  const total = [...arr1, ...arr2];
  total.sort((a,b) => a-b);
  return total[k-1]
};

처음 생각한 것은 배열을 합치고 정렬해준 다음에 k-1 인덱스 값을 뽑아내는 방식이다. 문제의 조건과는 맞지 않아서 테스트를 통과하지 못한다.

두번째 접근

const getItemFromTwoSortedArrays = function (arr1, arr2, k) {
  const m = arr1.length;
  const n = arr2.length; 

  const sorted = Array(m+n); // undefined로 이루어진 길이 m+n 배열
  
  let i= 0, j=0, idx= 0;

  while(i < m && j < n ) {
    if(arr1[i] < arr2[j]){
      sorted[idx++] = arr1[i++];
    } else {
      sorted[idx++] = arr2[j++];
    }
    console.log(sorted)
  }
  // sorted 배열의 끝까지 담기지 않으니까 추가로 while문을 돌려준다.
    while ( i < m) {
        sorted[idx++] = arr1[i++]
        console.log(sorted)
    }
    while(j<n) {
        sorted[idx++] = arr2[j++]
        console.log(sorted)
    }
    
  return sorted[k-1]

};


정렬하면서 배열에 담는 방식으로 풀어봤는데 이 방법도 역시 조건은 만족하지 못하고 입력받는 배열의 크기가 늘어날 수록 처리시간이 길어진다..

레퍼런스 코드

일단 시간복잡도를 해결하지 못한 naive 코드인데 count변수가 k-1번째 target 값을 찾을때까지 반복한다. arr1 인덱스를 접근하는 변수 left, arr2 인덱스를 접근하는 변수 right를 선언해서 값을 비교하면서 left, right 인덱스를 각각 증가시키면서 순회한다.

const getItemFromTwoSortedArrays = function (arr1, arr2, k) {
  let count = 0,
    left = 0,
    right = 0;
  let target;
  while (count < k) {
    if (arr1[left] < arr2[right]) {
      target = arr1[left];
      left++;
    } else {
      target = arr2[right];
      right++;
    }
    count++;
  }
  return target;
};

아래는 시간복잡도를 해결한 레퍼런스 코드인데, 이해가 잘 되지 않아서 다시 보는걸로!!

const getItemFromTwoSortedArrays = function (arr1, arr2, k) {
  let leftIdx = 0,
    rightIdx = 0;

  while (k > 0) {
    // 이진 탐색을 위해 각 배열에서 k를 절반으로 쪼개서 카운트 한다.
    let cnt = Math.ceil(k / 2);
    let leftStep = cnt,
      rightStep = cnt;

    // 엣지 케이스
    // 카운트가 남았음에도 배열의 끝에 도달하면 k를 나머지 배열쪽으로 넘긴다.
    if (leftIdx === arr1.length) {
      rightIdx = rightIdx + k;
      break;
    }

    if (rightIdx === arr2.length) {
      leftIdx = leftIdx + k;
      break;
    }

    // 엣지 케이스
    // 현재 카운트가 남아있는 후보 요소들보다 많을 경우, leftStep(현재 할당량)을 남아있는 요소들의 개수로 바꾼다.
    if (cnt > arr1.length - leftIdx) leftStep = arr1.length - leftIdx;
    if (cnt > arr2.length - rightIdx) rightStep = arr2.length - rightIdx;

    // 두 배열의 현재 검사 요소 위치를 비교해서, 그 값이 작은 배열은 비교한 위치 앞에 있는 요소들을 모두 후보군에서 제외시킨다.
    if (arr1[leftIdx + leftStep - 1] < arr2[rightIdx + rightStep - 1]) {
      leftIdx = leftIdx + leftStep;
      // 처리가 끝나면 k값을 절반으로 떨어뜨린다.
      k = k - leftStep;
    } else {
      rightIdx = rightIdx + rightStep;
      k = k - rightStep;
    }
  }

  leftMax = arr1[leftIdx - 1] || -1;
  rightMax = arr2[rightIdx - 1] || -1;

  return Math.max(leftMax, rightMax);
};
profile
Connecting dots

0개의 댓글