백준 온라인 저지, 이진탐색 / 3649번: 로봇프로젝트 (파이썬 / 백준 골드문제)

2021. 9. 15. 00:54알고리즘/이진탐색

728x90
반응형

문제

상근이와 선영이는 학교 숙제로 로봇을 만들고 있다. 로봇을 만들던 중에 구멍을 막을 두 레고 조각이 필요하다는 것을 깨달았다.

구멍의 너비는 x 센티미터이고, 구멍에 넣을 두 조각의 길이의 합은 구멍의 너비와 정확하게 일치해야 한다. 정확하게 일치하지 않으면, 프로젝트 시연을 할 때 로봇은 부수어질 것이고 상근이와 선영이는 F를 받게 된다. 구멍은 항상 두 조각으로 막아야 한다.

지난밤, 상근이와 선영이는 물리 실험실에 들어가서 레고 조각의 크기를 모두 정확하게 재고 돌아왔다. 구멍을 완벽하게 막을 수 있는 두 조각을 구하는 프로그램을 작성하시오.

입력

입력은 여러 개의 테스트 케이스로 이루어져 있다.

각 테스트 케이스의 첫째 줄에는 구멍의 너비 x (1 ≤ x ≤ 20, x는 정수)가 주어진다. x의 단위는 센티미터이다.

다음 줄에는 물리 실험실에 있는 레고 조각의 수 n이 주어진다. (0 ≤ n ≤ 1000000)

다음 n개의 줄에는 레고 조각의 길이 ℓ이 주어진다. ℓ은 양의 정수이며, 단위는 나노미터이다. 블록의 길이는 10 센티미터 (100000000 나노미터)를 넘지 않는다.

출력

각 테스트 케이스마다 한 줄에 하나씩, 구멍을 완벽하게 막을 수 있는 두 조각이 없다면 'danger'를 출력한다. 막을 수 있는 경우에는 'yes ℓ12'를 출력한다. (ℓ1 ≤ ℓ2)

정답이 여러 개인 경우에는 |ℓ1 - ℓ2|가 가장 큰 것을 출력한다.

예제 입력 1

1
4
9999998
1
2
9999999

예제 출력 1

yes 1 9999999
W3sicHJvYmxlbV9pZCI6IjM2NDkiLCJwcm9ibGVtX2xhbmciOiIwIiwidGl0bGUiOiJcdWI4NWNcdWJkMDcgXHVkNTA0XHViODVjXHVjODFkXHVkMmI4IiwiZGVzY3JpcHRpb24iOiI8cD5cdWMwYzFcdWFkZmNcdWM3NzRcdWM2NDAgXHVjMTIwXHVjNjAxXHVjNzc0XHViMjk0IFx1ZDU1OVx1YWQ1MCBcdWMyMTlcdWM4MWNcdWI4NWMgXHViODVjXHViZDA3XHVjNzQ0IFx1YjljY1x1YjRlNFx1YWNlMCBcdWM3ODhcdWIyZTQuIFx1Yjg1Y1x1YmQwN1x1Yzc0NCBcdWI5Y2NcdWI0ZTRcdWIzNTggXHVjOTExXHVjNWQwIFx1YWQ2Y1x1YmE0ZFx1Yzc0NCBcdWI5YzlcdWM3NDQgXHViNDUwIFx1YjgwOFx1YWNlMCBcdWM4NzBcdWFjMDFcdWM3NzQgXHVkNTQ0XHVjNjk0XHVkNTU4XHViMmU0XHViMjk0IFx1YWM4M1x1Yzc0NCBcdWFlNjhcdWIyZWNcdWM1NThcdWIyZTQuPFwvcD5cclxuXHJcbjxwPlx1YWQ2Y1x1YmE0ZFx1Yzc1OCBcdWIxMDhcdWJlNDRcdWIyOTQgeCBcdWMxM2NcdWQyZjBcdWJiZjhcdWQxMzBcdWM3NzRcdWFjZTAsIFx1YWQ2Y1x1YmE0ZFx1YzVkMCBcdWIxMjNcdWM3NDQgXHViNDUwIFx1Yzg3MFx1YWMwMVx1Yzc1OCBcdWFlMzhcdWM3NzRcdWM3NTggXHVkNTY5XHVjNzQwIFx1YWQ2Y1x1YmE0ZFx1Yzc1OCBcdWIxMDhcdWJlNDRcdWM2NDAgXHVjODE1XHVkNjU1XHVkNTU4XHVhYzhjIFx1Yzc3Y1x1Y2U1OFx1ZDU3NFx1YzU3YyBcdWQ1NWNcdWIyZTQuIFx1YzgxNVx1ZDY1NVx1ZDU1OFx1YWM4YyBcdWM3N2NcdWNlNThcdWQ1NThcdWM5YzAgXHVjNTRhXHVjNzNjXHViYTc0LCBcdWQ1MDRcdWI4NWNcdWM4MWRcdWQyYjggXHVjMmRjXHVjNWYwXHVjNzQ0IFx1ZDU2MCBcdWI1NGMgXHViODVjXHViZDA3XHVjNzQwIFx1YmQ4MFx1YzIxOFx1YzViNFx1YzljOCBcdWFjODNcdWM3NzRcdWFjZTAgXHVjMGMxXHVhZGZjXHVjNzc0XHVjNjQwIFx1YzEyMFx1YzYwMVx1Yzc3NFx1YjI5NCBGXHViOTdjIFx1YmMxYlx1YWM4YyBcdWI0MWNcdWIyZTQuIFx1YWQ2Y1x1YmE0ZFx1Yzc0MCBcdWQ1NmRcdWMwYzEgXHViNDUwIFx1Yzg3MFx1YWMwMVx1YzczY1x1Yjg1YyBcdWI5YzlcdWM1NDRcdWM1N2MgXHVkNTVjXHViMmU0LjxcL3A+XHJcblxyXG48cD5cdWM5YzBcdWIwOWNcdWJjMjQsIFx1YzBjMVx1YWRmY1x1Yzc3NFx1YzY0MCBcdWMxMjBcdWM2MDFcdWM3NzRcdWIyOTQgXHViYjNjXHViOWFjIFx1YzJlNFx1ZDVkOFx1YzJlNFx1YzVkMCBcdWI0ZTRcdWM1YjRcdWFjMDBcdWMxMWMgXHViODA4XHVhY2UwIFx1Yzg3MFx1YWMwMVx1Yzc1OCBcdWQwNmNcdWFlMzBcdWI5N2MgXHViYWE4XHViNDUwIFx1YzgxNVx1ZDY1NVx1ZDU1OFx1YWM4YyBcdWM3YWNcdWFjZTAgXHViM2NjXHVjNTQ0XHVjNjU0XHViMmU0LiBcdWFkNmNcdWJhNGRcdWM3NDQgXHVjNjQ0XHViY2JkXHVkNTU4XHVhYzhjIFx1YjljOVx1Yzc0NCBcdWMyMTggXHVjNzg4XHViMjk0IFx1YjQ1MCBcdWM4NzBcdWFjMDFcdWM3NDQgXHVhZDZjXHVkNTU4XHViMjk0IFx1ZDUwNFx1Yjg1Y1x1YWRmOFx1YjdhOFx1Yzc0NCBcdWM3OTFcdWMxMzFcdWQ1NThcdWMyZGNcdWM2MjQuPFwvcD5cclxuIiwiaW5wdXQiOiI8cD5cdWM3ODVcdWI4MjVcdWM3NDAgXHVjNWVjXHViN2VjIFx1YWMxY1x1Yzc1OCBcdWQxNGNcdWMyYTRcdWQyYjggXHVjZjAwXHVjNzc0XHVjMmE0XHViODVjIFx1Yzc3NFx1YjhlOFx1YzViNFx1YzgzOCBcdWM3ODhcdWIyZTQuPFwvcD5cclxuXHJcbjxwPlx1YWMwMSBcdWQxNGNcdWMyYTRcdWQyYjggXHVjZjAwXHVjNzc0XHVjMmE0XHVjNzU4IFx1Y2NhYlx1YzlmOCBcdWM5MDRcdWM1ZDBcdWIyOTQgXHVhZDZjXHViYTRkXHVjNzU4IFx1YjEwOFx1YmU0NCB4ICgxICZsZTsgeCAmbGU7IDIwLCB4XHViMjk0IFx1YzgxNVx1YzIxOClcdWFjMDAgXHVjOGZjXHVjNWI0XHVjOWM0XHViMmU0LiB4XHVjNzU4IFx1YjJlOFx1YzcwNFx1YjI5NCBcdWMxM2NcdWQyZjBcdWJiZjhcdWQxMzBcdWM3NzRcdWIyZTQuPFwvcD5cclxuXHJcbjxwPlx1YjJlNFx1Yzc0YyBcdWM5MDRcdWM1ZDBcdWIyOTQgXHViYjNjXHViOWFjIFx1YzJlNFx1ZDVkOFx1YzJlNFx1YzVkMCBcdWM3ODhcdWIyOTQgXHViODA4XHVhY2UwIFx1Yzg3MFx1YWMwMVx1Yzc1OCBcdWMyMTggblx1Yzc3NCBcdWM4ZmNcdWM1YjRcdWM5YzRcdWIyZTQuICgwICZsZTsgbiAmbGU7IDEwMDAwMDApPFwvcD5cclxuXHJcbjxwPlx1YjJlNFx1Yzc0YyBuXHVhYzFjXHVjNzU4IFx1YzkwNFx1YzVkMFx1YjI5NCBcdWI4MDhcdWFjZTAgXHVjODcwXHVhYzAxXHVjNzU4IFx1YWUzOFx1Yzc3NCBcdTIxMTNcdWM3NzQgXHVjOGZjXHVjNWI0XHVjOWM0XHViMmU0LiBcdTIxMTNcdWM3NDAgXHVjNTkxXHVjNzU4IFx1YzgxNVx1YzIxOFx1Yzc3NFx1YmE3MCwmbmJzcDtcdWIyZThcdWM3MDRcdWIyOTQgXHViMDk4XHViMTc4XHViYmY4XHVkMTMwXHVjNzc0XHViMmU0LiBcdWJlMTRcdWI4NWRcdWM3NTggXHVhZTM4XHVjNzc0XHViMjk0IDEwIFx1YzEzY1x1ZDJmMFx1YmJmOFx1ZDEzMCAoMTAwMDAwMDAwIFx1YjA5OFx1YjE3OFx1YmJmOFx1ZDEzMClcdWI5N2MgXHViMTE4XHVjOWMwIFx1YzU0YVx1YjI5NFx1YjJlNC48XC9wPlxyXG4iLCJvdXRwdXQiOiI8cD5cdWFjMDEgXHVkMTRjXHVjMmE0XHVkMmI4IFx1Y2YwMFx1Yzc3NFx1YzJhNFx1YjljOFx1YjJlNCBcdWQ1NWMgXHVjOTA0XHVjNWQwIFx1ZDU1OFx1YjA5OFx1YzUyOSwgXHVhZDZjXHViYTRkXHVjNzQ0IFx1YzY0NFx1YmNiZFx1ZDU1OFx1YWM4YyBcdWI5YzlcdWM3NDQgXHVjMjE4IFx1Yzc4OFx1YjI5NCBcdWI0NTAgXHVjODcwXHVhYzAxXHVjNzc0IFx1YzVjNlx1YjJlNFx1YmE3NCAmIzM5O2RhbmdlciYjMzk7XHViOTdjIFx1Y2Q5Y1x1YjgyNVx1ZDU1Y1x1YjJlNC4gXHViOWM5XHVjNzQ0IFx1YzIxOCBcdWM3ODhcdWIyOTQgXHVhY2JkXHVjNmIwXHVjNWQwXHViMjk0ICYjMzk7eWVzIFx1MjExMzxzdWI+MTxcL3N1Yj4gXHUyMTEzPHN1Yj4yPFwvc3ViPiYjMzk7XHViOTdjIFx1Y2Q5Y1x1YjgyNVx1ZDU1Y1x1YjJlNC4gKFx1MjExMzxzdWI+MTxcL3N1Yj4gJmxlOyBcdTIxMTM8c3ViPjI8XC9zdWI+KTxcL3A+XHJcblxyXG48cD5cdWM4MTVcdWIyZjVcdWM3NzQgXHVjNWVjXHViN2VjIFx1YWMxY1x1Yzc3OCBcdWFjYmRcdWM2YjBcdWM1ZDBcdWIyOTQgfFx1MjExMzxzdWI+MTxcL3N1Yj4gLSBcdTIxMTM8c3ViPjI8XC9zdWI+fFx1YWMwMCBcdWFjMDBcdWM3YTUgXHVkMDcwIFx1YWM4M1x1Yzc0NCBcdWNkOWNcdWI4MjVcdWQ1NWNcdWIyZTQuPFwvcD5cclxuIiwiaGludCI6IiIsIm9yaWdpbmFsIjoiMCIsImh0bWxfdGl0bGUiOiIwIiwicHJvYmxlbV9sYW5nX3Rjb2RlIjoiS29yZWFuIn0seyJwcm9ibGVtX2lkIjoiMzY0OSIsInByb2JsZW1fbGFuZyI6IjEiLCJ0aXRsZSI6IkpvaW50IFZlbnR1cmUiLCJkZXNjcmlwdGlvbiI6IjxwPkxpZXNiZXRoIGFuZCBKYW4gYXJlIGJ1aWxkaW5nIGEgcm9ib3QgZm9yIGEgY291cnNlIHByb2plY3QgYW5kIGhhdmUgZGlzY292ZXJlZCB0aGF0IHRoZXkgbmVlZCB0byBcdWZiMDF0IHR3byBwaWVjZXMgb2YgTGVnbyBpbnRvIGFuIG9wZW5pbmcuPFwvcD5cclxuXHJcbjxwPlRoZSBvcGVuaW5nIGlzIHggY2VudGltZXRyZXMgd2lkZSBhbmQgdGhlIHN1bSBvZiB0aGUgbGVuZ3RocyBvZiB0aGUgdHdvIHBpZWNlcyBoYXMgdG8gYmUgcHJlY2lzZWx5IGVxdWFsIHRvIHRoZSB3aWR0aCBvZiB0aGUgb3BlbmluZywgb3IgZWxzZSB0aGUgcm9ib3Qgd2lsbCBicmVhayBkdXJpbmcgdGhlIHByb2plY3QgZGVtb25zdHJhdGlvbiwgd2l0aCBjYXRhc3Ryb3BoaWMgY29uc2VxdWVuY2VzIGZvciB0aGUgZ3JhZGVzIG9mIHRoZSB0d28gc3R1ZGVudHMuPFwvcD5cclxuXHJcbjxwPkx1Y2tpbHksIExpZXNiZXRoIGFuZCBKYW4gd2VyZSBhYmxlIHRvIHNuZWFrIGludG8gdGhlIHBoeXNpY3MgbGFib3JhdG9yeSBsYXRlIG9uZSBuaWdodCB0byBtZWFzdXJlIHRoZSBsZW5ndGhzIG9mIHRoZWlyIHJlbWFpbmluZyBMZWdvIHBpZWNlcyB2ZXJ5IGFjY3VyYXRlbHkuIE5vdyB0aGV5IGp1c3QgbmVlZCB0byBzZWxlY3QgdHdvIHBpZWNlcyB0aGF0IHdpbGwgXHVmYjAxdCB0aGUgb3BlbmluZyBwZXJmZWN0bHkuPFwvcD5cclxuIiwiaW5wdXQiOiI8cD5UaGUgaW5wdXQgY29uc2lzdHMgbXVsdGlwbGUgdGVzdCBjYXNlcy48XC9wPlxyXG5cclxuPHA+Rm9yIGVhY2ggdGVzdCBjYXNlLCB5b3UgZ2V0OjxcL3A+XHJcblxyXG48dWw+XHJcblx0PGxpPmEgbGluZSBjb250YWluaW5nIG9uZSBwb3NpdGl2ZSBpbnRlZ2VyOiB4LCBkZW5vdGluZyB0aGUgd2lkdGggb2YgdGhlIG9wZW5pbmcgaW4gY2VudGltZXRyZXMsIHdpdGggMSAmbGU7IHggJmxlOyAyMC48XC9saT5cclxuXHQ8bGk+YSBsaW5lIGNvbnRhaW5pbmcgb25lIG5vbi1uZWdhdGl2ZSBpbnRlZ2VyOiBuLCBkZW5vdGluZyB0aGUgcmVtYWluaW5nIG51bWJlciBvZiBMZWdvIHBpZWNlcyBMaWVzYmV0aCBhbmQgSmFuIGhhdmUgYWNjZXNzIHRvLCB3aXRoIDAgJmxlOyBuICZsZTsgMTAwMDAwMC48XC9saT5cclxuXHQ8bGk+biBsaW5lcyBjb250YWluaW5nIHBvc2l0aXZlIGludGVnZXJzIFx1MjExMywgZGVub3RpbmcgbGVuZ3RocyBvZiBMZWdvIHBpZWNlcyBpbiBuYW5vbWV0cmVzLiBMaWVzYmV0aCBhbmQgSmFuIGhhdmUgdG9sZCB5b3UgdGhhdCBubyBwaWVjZSBvZiBMZWdvIGlzIGxvbmdlciB0aGFuIDEwIGNlbnRpbWV0cmVzLCBvciAxMDAwMDAwMDAgbmFub21ldHJlcy48XC9saT5cclxuPFwvdWw+XHJcbiIsIm91dHB1dCI6IjxwPkZvciBlYWNoIHRlc3QgY2FzZSwgYSByb3cgY29udGFpbmluZyB0aGUgd29yZCAmbHNxdW87ZGFuZ2VyJnJzcXVvOyBpZiBubyB0d28gcGllY2VzIG9mIExlZ28gZXhpc3QgdGhhdCBwcmVjaXNlbHkgXHVmYjAxdCBpbnRvIHRoZSBvcGVuaW5nLCBvciAmbHNxdW87eWVzIFx1MjExMzxzdWI+MTxcL3N1Yj4gXHUyMTEzPHN1Yj4yPFwvc3ViPiZyc3F1bzssIHdpdGggXHUyMTEzPHN1Yj4xPFwvc3ViPiAmbGU7IFx1MjExMzxzdWI+MjxcL3N1Yj4sIHNob3VsZCB0d28gc3VjaCBwaWVjZXMgb2YgbGVuZ3RocyBcdTIxMTM8c3ViPjE8XC9zdWI+IGFuZCBcdTIxMTM8c3ViPjI8XC9zdWI+IGV4aXN0LjxcL3A+XHJcblxyXG48cD5JbiBjYXNlIG11bHRpcGxlIHNvbHV0aW9ucyBleGlzdCwgYSBzb2x1dGlvbiBtYXhpbWlzaW5nIHRoZSBzaXplIGRpZmZlcmVuY2UgfFx1MjExMzxzdWI+MTxcL3N1Yj4gLSBcdTIxMTM8c3ViPjI8XC9zdWI+fCBtdXN0IGJlIHByaW50ZWQuPFwvcD5cclxuIiwiaGludCI6IiIsIm9yaWdpbmFsIjoiMSIsImh0bWxfdGl0bGUiOiIwIiwicHJvYmxlbV9sYW5nX3Rjb2RlIjoiRW5nbGlzaCJ9XQ==

접근 방법

- 주어진 레고의 길이 리스트를 주어진 x의 절반을 기준으로 나누어 리스트(short_lego, long_lego)에 저장한 뒤, 이를 각각 오름차순 정렬한다.
- 단 정확히 x의 절반이 되는 레고가 두 개라면 해당 레고는 바로 정답 리스트(result)에 저장한다.
- 이후 short_lego와 long_lego의 길이가 더 짧은 걸 기준으로 하나씩 탐색한다.(최적화를 위해) 그리고 나머지 레고에 대해 이진탐색을 진행하며 탐색 중인 두 길이가 x와 같다면 정답 리스트(result)에 저장해준다.
- 모든 탐색이 끝나고 result를 확인하며 abs(x[0] - x[1])이 가장 큰 값을 출력한다.
- 단위: 10**7 나노미터 = 1cm

코드

# 접근방법
# 주어진 레고의 길이 리스트를 주어진 x의 절반을 기준으로 나누어 리스트(short_lego, long_lego)에 저장한 뒤, 이를 각각 오름차순 정렬한다.
# 단 정확히 x의 절반이 되는 레고가 두 개라면 해당 레고는 바로 정답 리스트(result)에 저장한다.
# 이후 short_lego와 long_lego의 길이가 더 짧은 걸 기준으로 하나씩 탐색한다.(최적화를 위해) 그리고 나머지 레고에 대해 이진탐색을 진행하며 탐색 중인 두 길이가 x와 같다면 정답 리스트(result)에 저장해준다.
# 모든 탐색이 끝나고 result를 확인하며 abs(x[0] - x[1])이 가장 큰 값을 출력한다.
# 단위: 10**7 나노미터 = 1cm


import sys
input = sys.stdin.readline

while True:
    try:
        x = int(input())
        n = int(input())
        lego = [int(input()) for _ in range(n)]
        x_ = x * (10 ** 7)
        short_lego = []
        mid_lego = []
        long_lego = []
        result = []

         
        for l in lego:
            if l > x_ // 2:    
                long_lego.append(l)
            elif l < x_ // 2:
                short_lego.append(l)
            else:
                result.append([l, l])
    
        long_lego.sort()
        short_lego.sort()

        if len(short_lego) <= len(long_lego):
            for lego1 in short_lego:
                start = 0
                end = len(long_lego) - 1
                while start <= end:
                    mid = (start + end) // 2
                    lego2 = long_lego[mid]
            
                    if lego1 + lego2 < x_:
                        start = mid + 1
                    elif lego1 + lego2 > x_:
                        end = mid - 1
                    else:
                        result.append([lego1, lego2])
                        break
            
        else:
            for lego1 in long_lego:
                start = 0
                end = len(short_lego) - 1
                while start <= end:
                    mid = (start + end) // 2
                    lego2 = short_lego[mid]
            
                    if lego1 + lego2 < x_:
                        start = mid + 1
                    elif lego1 + lego2 > x_:
                        end = mid - 1
                    else:
                        result.append([lego2, lego1])
                        break
    
        result.sort(key=lambda x: abs(x[1] - x[0]), reverse = True)
        if result:
            print('yes', result[0][0], result[0][1])
        else:
            print('danger')
    except:
        break
728x90
반응형