[Greedy] 11000번 - 강의실 배정 (13일차)

bob.sort·2021년 6월 13일
0
post-thumbnail
post-custom-banner

C/C++식 코드

#코드 실행시간 단축을 위한 sys 모듈 import
import sys
input = sys.stdin.readline
#강의의 개수 n개 입력
n = int(input())
#각 강의의 시작시간과 끝시간 입력
course = [list(map(int, input().split())) for _ in range(n)]
#강의들을 시작시작을 기준으로 정렬
course.sort(key = lambda x: x[0])
#각 강의실에서 사용되고 있는 강의의 끝나는 시간을 저장하는 리스트
room = [0]*n
#필요한 강의의 수를 저장하는 변수
cnt = 0
#모든 강의의 수 n만큼 탐색
for i in range(n):
    #지칭하고 있는 강의가 가장 첫번째로 시작하는 강의일때
    if(i == 0):
        #해당 강의를 첫번째 강의실에 배정
        room[i] = course[i][1]
        #필요한 강의실의 수를 추가
        cnt += 1
    #지칭하고 있는 강의가 가장 첫번째로 시작하지 않을 때
    else:
        #사용되고 있는 모든 강의실의 수 cnt만큼 탐색
        for j in range(cnt):
            #현재 지칭하고 있는 강의의 시작시간이 지칭하고 있는 강의실의 끝나는 시간과 같거나 그 뒤일 때
            if(course[i][0] >= room[j]):
                #해당 강의실에 지칭하고 있는 강의를 배정(끝나는 시간을 업데이트)
                room[j] = course[i][1]
                #더이상의 탐색은 의미가 없으니 탐색 종료
                break
            #마지막 강의실까지 탐색을 했을 때, 해당 강의가 어떤 강의실에도 배정되지 않은 경우
            elif(j == cnt-1):
                #강의실을 하나 추가하고, 해당 강의실에 해당 강의를 배정
                room[cnt] = course[i][1]
                #필요한 강의실의 수를 추가
                cnt += 1
#모든 강의가 강의실에 배정되었을때, 필요한 강의실의 수를 출력
print(cnt)

#인사이트
#Passport Control에 사용된 큐를 응용한 로직
#각 강의실을 각각 큐로 사용하여, 큐에 배정된 강의의 강의종료시간을 저장하고,
#각 강의를 탐색하면서 강의실에 저장된 강의종료시간과 같거나 늦게 시작한다면, 해당 강의실의 강의종료시간을 업데이트
#순차탐색을 이용해 C의 로직으로 작성된 코드이나, 파이썬 특유의 느린 실행시간에 의해 시간초과가 발생
#해당 로직은 C로 재작성할 경우 더욱 효율적인 코드
#각 언어에 알맞는 코드 로직을 짤 필요가 있음을 깨달았다

Pythonic 코드

#min heap을 사용하기 위한 heapq 모듈 import
import heapq
#코드 실행 시간 단축을 위한 sys 모듈 import
import sys
#강의실의 수 n 입력
N = int(input())
#각 강의의 시작시간과 끝시간 입력
course = [list(map(int,sys.stdin.readline().split())) for _ in range(N)]
#강의들을 시작시작을 기준으로 정렬
course.sort(key=lambda x: x[0])
#각 강의실에서 사용되고 있는 강의의 끝나는 시간을 저장하는 리스트
room = []
#첫번째 강의실에 가장 먼저 시작하는 강의를 배정(해당 강의가 끝나는 시간을 저장)
heapq.heappush(room,course[0][1])
#2번째 강의부터 마지막 강의까지
for i in range(1,N):
    #강의가 가장 먼저 끝나는 강의실보다 해당 강의의 시작시간이 빠를 때(해당 강의의 시작시간에 모든 강의실이 사용중일 때)
    if room[0] > course[i][0]:
        #해당 강의를 새로운 강의실에 배정
        heapq.heappush(room,course[i][1])
    #강의가 가장 먼저 끝나는 강의실보다 해당 강의의 시작시간이 같거나 늦을 때(해당 강의실에서 해당 강의를 다음 강의로 사용가능할 때)
    else:
        #해당 강의실에 저장된 강의종료시간을 삭제하고
        heapq.heappop(room)
        #해당 강의실에 해당 강의를 배정 (강의종료시간을 업데이트)
        #모든 강의실에 저장된 강의종료시간을 가장 빨리 끝나는 시간을 기준으로 min heap 정렬한다
        heapq.heappush(room,course[i][1])
#사용되고 있는 강의의 수를 저장
print(len(room))

#인사이트
#핵심은 얼마나 적은 탐색으로 강의가 배정되기에 적합한 강의실을 찾을 수 있는가이다
#min heap의 경우, O(nlogn)으로 탐색하기 때문에 적은 탐색으로 강의실을 찾을 수 있다
#pythonic한 코딩은 로직을 코드로 구현하는 것을 넘어 적절한 모듈을 적재적소에 활용하는 것이다
profile
Interest in Computer Graphics and Computer Vision
post-custom-banner

0개의 댓글