이숭간 공부기록

[백준] 21608번 파이썬 _ 상어 초등학교 (삼성SW) 본문

알고리즘/백준

[백준] 21608번 파이썬 _ 상어 초등학교 (삼성SW)

이숭간 2021. 7. 19. 15:19
728x90

https://www.acmicpc.net/problem/21608

 

21608번: 상어 초등학교

상어 초등학교에는 교실이 하나 있고, 교실은 N×N 크기의 격자로 나타낼 수 있다. 학교에 다니는 학생의 수는 N2명이다. 오늘은 모든 학생의 자리를 정하는 날이다. 학생은 1번부터 N2번까지 번호

www.acmicpc.net

문제유형 : 구현(시뮬레이션), 완전탐색 

 

문제풀이 :

 

한명의 학생이 자리를 찾을때, 비어있는 모든 자리 탐색하며 조건에 가장 부합한 자리를 찾아야한다. (완전탐색)

  • 학생은 딕셔너리형태로 {학생넘버 : 좋아하는학생리스트} 형태로 입력받는다.
  • check(j,k)함수 : j,k인덱스 자리에서 인접한 자리에 위치한 친구들리스트와 빈칸의수를 리턴하는 함수 
    • 한명의 학생이 앉을 자리를 선택하는 과정에서 빈자리를 모두 탐색해야하므로, 이때 빈칸인 j,k에 대해서 check함수를 호출
  • find_seat(l)함수 : 자리를 결정하는 함수
    • l배열의 원소는 다음과 같다. (좋아하는 친구들수, 빈칸의수, 행, 열 )
    • 이때 좋아하는 친구들이 많은 순 -> 빈칸이 많은 순 -> 행이 작은순 -> 열이 작은순 정렬해야하는데 sorted()함수를 통해서 한번에 조건 4개를모두 따지기위해서 (내림차순) 좋아하는 친구들수와 빈칸의수를 음수로해서 넣는다.
    • 이렇게 하면 내림차순 정렬했을때 한번에 위 조건에 가장 부합하는 원소가 맨앞에 오게된다.
  • table배열 : 누군가 앉았다면 False에서 해당학생의 번호로 바꿔준다.
    • 앉을 자리를 선택할때 table에서 False인 위치만 체크한다.

 

 

정답코드 :

import sys
from collections import defaultdict
input = sys.stdin.readline

n = int(input())

dx = [-1,1,0,0]
dy = [0,0,-1,1]

def check(j,k):
    near_list = []
    empty = 0
    for i in range(4):
        if j+dx[i] >=0 and j+dx[i] < n and k+dy[i] >=0 and k+dy[i] < n:
            if table[j+dx[i]][k+dy[i]]:
                near_list.append(table[j+dx[i]][k+dy[i]])
            else:
                empty += 1

    return near_list,empty

def find_seat(l):
    l = sorted(l, key=lambda x:(x[0], x[1], x[2], x[3]))
    cnt,emp,x,y = l[0]
    return x,y


student = defaultdict(list)

for _ in range(n*n):
    temp = list(map(int, input().split()))
    student[temp[0]] = temp[1:]

table = [[False] * n for _ in range(n)]

for i in student.keys():
    l = []
    temp =[]
    cnt = 0
    like_friend = 0 # i가 앉을수있는 위치에서 좋아하는 친구들의 수
    for j in range(n):
        for k in range(n):
            if table[j][k]: # 누가 앉아있으면 건너뛰고
                continue
            #i학생이 j,k에 앉았을때 인접한 친구들과 인접한 빈칸의 수
            near_list, emp = check(j,k)

            for like in student[i]:
                if like in near_list:
                    cnt += 1
            temp.append(-cnt)
            temp.append(-emp)
            temp.append(j)
            temp.append(k)
            l.append(temp)
            temp = []
            cnt = 0

    #print(l) # l- 친구리스트와 빈칸, 해당 자리
    x,y = find_seat(l)
    table[x][y] = i

answer = 0
score = {0:0, 1:1, 2:10, 3:100, 4:1000}
for i in range(n):
    for j in range(n):
        count = 0
        friend_list = check(i,j)[0]
        for f in friend_list:
            if f in student[table[i][j]]:
                count += 1
        answer += score[count]

print(answer)

걸린시간 : 1시간