HyunsooZo's TIL logo HyunsooZo's TIL

알고리즘

이분탐색이란?

이분탐색이란, 정렬된 배열에서 특정 값을 찾는 탐색 알고리즘이다. 주어진 배열을 반으로 나누어 탐색 범위를 좁혀가는 방식으로 동작한다.
매 단계마다 탐색 범위를 절반으로 줄여나가므로 배열의 크기에 대해 로그 시간 복잡도를 가진다.
배열을 정렬하는 데에 O(LogN) 시간 복잡도를 가지고
이분탐색을 사용하여 값을 찾는 과정 또한 로그 시간 복잡도로 이루어지며 중간을 기준으로
탐색대상을 절반씩 줄여가므로 전체적으로 O(LogN)의 시간 복잡도를 가지게 된다.
이는 완전탐색의 O(N)에 비하면 매우 빠른속도이다.
O(logN)만에 값을 찾을 수 있는 이유는 중간을 기준으로 탐색 대상을 절반씩 줄여나가기 때문이다.
아래는 이해를 돕기위해 간단한 그림으로 배열 속에서 이분탐색을 통해 값을 찾아나가는 과정을 그려보았다.

예시

알고리즘 기본 예시문제

예시

프로그래머스 문제

프로그래머스 Lv2. 시소짝꿍

어느 공원 놀이터에는 시소가 하나 설치되어 있습니다. 이 시소는 중심으로부터 2(m), 3(m), 4(m) 거리의 지점에 좌석이 하나씩 있습니다.
이 시소를 두 명이 마주 보고 탄다고 할 때, 시소가 평형인 상태에서 각각에 의해 시소에 걸리는 토크의 크기가 서로 상쇄되어 완전한 균형을 이룰 수 있다면 그 두 사람을 시소 짝꿍이라고 합니다.
즉, 탑승한 사람의 무게와 시소 축과 좌석 간의 거리의 곱이 양쪽 다 같다면 시소 짝꿍이라고 할 수 있습니다.
사람들의 몸무게 목록 weights이 주어질 때, 시소 짝꿍이 몇 쌍 존재하는지 구하여 return 하도록 solution 함수를 완성해주세요.

문제 분석

문제분석

문제 제한사항에서 배열의 길이가 10,000이라고 주어졌으니
완전탐색을 돌리면 10억이 넘아가므로 성능테스트에 걸릴것으로 예상.
이분탐색을 돌리면 될것이다.
일단 배열 정렬 후 만족조건을 생각해보면
1m짜리 자리가 없기때문에 아래와 같은 4개 조건만 만족하면된다.
1. a == b 일것 (이는 곧 a*n==b*n, 즉1:1)
2. a*4 == b*2 일것 (이는 곧 a*2 == b, 즉 2:1)
3. a*3 == b*2 일것 (즉 3:2)
4. a*4 == b*3 일것 (즉 4:3)

과 같다. a는 언제나 b 보다 작으므로 반대의 경우는 굳이 고려할 필요가 없다. (예 : a*1 == b*2 는 항상 거짓이다.)

문제풀이

위 조건에 따라 이분탐색을 진행하면 아래와 같다.

import java.util.Arrays;

public class Solution {
    // 전역 변수 선언(파라메터 수 줄이기위함)
    static long answer = 0; // 정답을 저장하는 변수
    static int[] arr; // 입력 weights 배열을 저장하는 배열
    static int len; // weights 배열의 길이를 저장하는 변수

    // 주어진 weights 배열을 이용하여 문제를 해결하는 메서드
    public long solution(int[] weights) {
        arr = weights; // 입력 weights 배열을 전역 배열 arr에 할당
        len = arr.length; // weights 배열의 길이를 len 변수에 저장
        Arrays.sort(arr); // weights 배열을 오름차순으로 정렬

        // weights 배열의 모든 요소에 대해 반복
        for (int i = 0; i < len; i++) {
            findSameValues(i); // 현재 값과 동일한 값이 있는지 찾는 메서드 호출

            // 현재 값의 두 배가 존재하는지 확인하고, 있다면 answer 증가
            if (binarySearch(arr[i] * 2)) {
                answer++;
            }

            // 현재 값이 3의 배수인지 확인
            if (arr[i] % 3 == 0) {
                // 현재 값의 두 배를 3으로 나눈 값이 존재하는지 확인하고, 있다면 answer 증가
                if (binarySearch(arr[i] * 2 / 3)) {
                    answer++;
                }
                // 현재 값의 네 배를 3으로 나눈 값이 존재하는지 확인하고, 있다면 answer 증가
                if (binarySearch(arr[i] * 4 / 3)) {
                    answer++;
                }
            }
        }
        return answer; // 최종 결과 반환
    }

    // 현재 값과 동일한 값이 있는지 찾는 메서드
    void findSameValues(int val) {
        for (int i = val + 1; i < len; i++) {
            // 현재 값과 동일한 값을 찾으면 answer 증가하고, 다른 값이 나오면 반복문 종료
            if (arr[i] == arr[val]) {
                answer++;
            } else {
                break;
            }
        }
    }

    // 이진 탐색을 통해 target 값이 배열에 존재하는지 확인하는 메서드
    boolean binarySearch(int target) {
        int left = 0; // 배열의 가장 왼쪽 인덱스
        int right = len - 1; // 배열의 가장 오른쪽 인덱스

        // 이진 탐색 수행
        while (left <= right) {
            int mid = left + (right - left) / 2; // 중간 위치 계산

            // 중간 위치의 값이 타겟 값과 일치할 때
            if (arr[mid] == target) {
                // 중복된 값이 있을 수 있으므로, 타겟 값의 좌우로 같은 값이 있는지 확인하고 answer 증가
                for (int i = mid + 1; i < len; i++) {
                    if (arr[i] == target) {
                        answer++;
                    } else {
                        break;
                    }
                }
                for (int i = mid - 1; i >= 0; i--) {
                    if (arr[i] == target) {
                        answer++;
                    } else {
                        break;
                    }
                }
                return true; // 타겟 값이 존재함을 반환
            }
            // 중간 위치의 값보다 타겟 값이 큰 경우, 오른쪽 부분 탐색
            else if (arr[mid] < target) {
                left = mid + 1;
            }
            // 중간 위치의 값보다 타겟 값이 작은 경우, 왼쪽 부분 탐색
            else {
                right = mid - 1;
            }
        }

        return false; // 타겟 값이 존재하지 않음을 반환
    }
}

결과

TOP