백준 온라인 저지, 분리집합 / 20040번: 사이클게임 (파이썬 / 백준 골드문제)

2021. 9. 15. 00:52알고리즘/분리집합

728x90
반응형

문제

사이클 게임은 두 명의 플레이어가 차례대로 돌아가며 진행하는 게임으로, 선 플레이어가 홀수 번째 차례를, 후 플레이어가 짝수 번째 차례를 진행한다. 게임 시작 시 0 부터 n − 1 까지 고유한 번호가 부여된 평면 상의 점 n 개가 주어지며, 이 중 어느 세 점도 일직선 위에 놓이지 않는다. 매 차례 마다 플레이어는 두 점을 선택해서 이를 연결하는 선분을 긋는데, 이전에 그린 선분을 다시 그을 수는 없지만 이미 그린 다른 선분과 교차하는 것은 가능하다. 게임을 진행하다가 처음으로 사이클을 완성하는 순간 게임이 종료된다. 사이클 C는 플레이어가 그린 선분들의 부분집합으로, 다음 조건을 만족한다.

C에 속한 임의의 선분의 한 끝점에서 출발하여 모든 선분을 한 번씩만 지나서 출발점으로 되돌아올 수 있다.

문제는 선분을 여러 개 그리다 보면 사이클이 완성 되었는지의 여부를 판단하기 어려워 이미 사이클이 완성되었음에도 불구하고 게임을 계속 진행하게 될 수 있다는 것이다. 이 문제를 해결하기 위해서 게임의 진행 상황이 주어지면 몇 번째 차례에서 사이클이 완성되었는지, 혹은 아직 게임이 진행 중인지를 판단하는 프로그램을 작성하려 한다.

입력으로 점의 개수 nm 번째 차례까지의 게임 진행 상황이 주어지면 사이클이 완성 되었는지를 판단하고, 완성되었다면 몇 번째 차례에서 처음으로 사이클이 완성된 것인지를 출력하는 프로그램을 작성하시오.

입력

입력은 표준입력을 사용한다. 입력의 첫 번째 줄에는 점의 개수를 나타내는 정수 3 ≤ n ≤ 500,000 과 진행된 차례의 수를 나타내는 정수 3 ≤ m ≤ 1,000,000 이 주어진다. 게임에서 사용하는 n개의 점에는 0 부터 n − 1 까지 고유한 번호가 부여되어 있으며, 이 중 어느 세 점도 일직선 위에 놓이지 않는다. 이어지는 m 개의 입력 줄에는 각각 i번째 차례에 해당 플레이어가 선택한 두 점의 번호가 주어진다 (1 ≤ im).

출력

출력은 표준출력을 사용한다. 입력으로 주어진 케이스에 대해, m 번째 차례까지 게임을 진행한 상황에서 이미 게임이 종료되었다면 사이클이 처음으로 만들어진 차례의 번호를 양의 정수로 출력하고, m 번의 차례를 모두 처리한 이후에도 종료되지 않았다면 0을 출력한다.

예제 입력 1

6 5
0 1
1 2
2 3
5 4
0 4

예제 출력 1

0

예제 입력 2

6 5
0 1
1 2
1 3
0 3
4 5

예제 출력 2

4
[{"problem_id":"20040","problem_lang":"0","title":"\uc0ac\uc774\ud074 \uac8c\uc784","description":"<p>\uc0ac\uc774\ud074 \uac8c\uc784\uc740 \ub450 \uba85\uc758 \ud50c\ub808\uc774\uc5b4\uac00 \ucc28\ub840\ub300\ub85c \ub3cc\uc544\uac00\uba70 \uc9c4\ud589\ud558\ub294 \uac8c\uc784\uc73c\ub85c, \uc120 \ud50c\ub808\uc774\uc5b4\uac00 \ud640\uc218 \ubc88\uc9f8 \ucc28\ub840\ub97c, \ud6c4 \ud50c\ub808\uc774\uc5b4\uac00 \uc9dd\uc218 \ubc88\uc9f8 \ucc28\ub840\ub97c \uc9c4\ud589\ud55c\ub2e4. \uac8c\uc784 \uc2dc\uc791 \uc2dc 0 \ubd80\ud130 <em>n<\/em> &minus; 1 \uae4c\uc9c0 \uace0\uc720\ud55c \ubc88\ud638\uac00 \ubd80\uc5ec\ub41c \ud3c9\uba74 \uc0c1\uc758 \uc810 <em>n<\/em> \uac1c\uac00 \uc8fc\uc5b4\uc9c0\uba70, \uc774 \uc911 \uc5b4\ub290 \uc138 \uc810\ub3c4 \uc77c\uc9c1\uc120 \uc704\uc5d0 \ub193\uc774\uc9c0 \uc54a\ub294\ub2e4. \ub9e4 \ucc28\ub840 \ub9c8\ub2e4 \ud50c\ub808\uc774\uc5b4\ub294 \ub450 \uc810\uc744 \uc120\ud0dd\ud574\uc11c \uc774\ub97c \uc5f0\uacb0\ud558\ub294 \uc120\ubd84\uc744 \uae0b\ub294\ub370, \uc774\uc804\uc5d0 \uadf8\ub9b0 \uc120\ubd84\uc744 \ub2e4\uc2dc \uadf8\uc744 \uc218\ub294 \uc5c6\uc9c0\ub9cc \uc774\ubbf8 \uadf8\ub9b0 \ub2e4\ub978 \uc120\ubd84\uacfc \uad50\ucc28\ud558\ub294 \uac83\uc740 \uac00\ub2a5\ud558\ub2e4. \uac8c\uc784\uc744 \uc9c4\ud589\ud558\ub2e4\uac00 \ucc98\uc74c\uc73c\ub85c \uc0ac\uc774\ud074\uc744 \uc644\uc131\ud558\ub294 \uc21c\uac04 \uac8c\uc784\uc774 \uc885\ub8cc\ub41c\ub2e4. \uc0ac\uc774\ud074 <em>C<\/em>\ub294 \ud50c\ub808\uc774\uc5b4\uac00 \uadf8\ub9b0 \uc120\ubd84\ub4e4\uc758 \ubd80\ubd84\uc9d1\ud569\uc73c\ub85c, \ub2e4\uc74c \uc870\uac74\uc744 \ub9cc\uc871\ud55c\ub2e4.<\/p>\r\n\r\n<blockquote>\r\n<p><em>C<\/em>\uc5d0 \uc18d\ud55c \uc784\uc758\uc758 \uc120\ubd84\uc758 \ud55c \ub05d\uc810\uc5d0\uc11c \ucd9c\ubc1c\ud558\uc5ec \ubaa8\ub4e0 \uc120\ubd84\uc744 \ud55c \ubc88\uc529\ub9cc \uc9c0\ub098\uc11c \ucd9c\ubc1c\uc810\uc73c\ub85c \ub418\ub3cc\uc544\uc62c \uc218 \uc788\ub2e4.<\/p>\r\n<\/blockquote>\r\n\r\n<p>\ubb38\uc81c\ub294 \uc120\ubd84\uc744 \uc5ec\ub7ec \uac1c \uadf8\ub9ac\ub2e4 \ubcf4\uba74 \uc0ac\uc774\ud074\uc774 \uc644\uc131 \ub418\uc5c8\ub294\uc9c0\uc758 \uc5ec\ubd80\ub97c \ud310\ub2e8\ud558\uae30 \uc5b4\ub824\uc6cc \uc774\ubbf8 \uc0ac\uc774\ud074\uc774 \uc644\uc131\ub418\uc5c8\uc74c\uc5d0\ub3c4 \ubd88\uad6c\ud558\uace0 \uac8c\uc784\uc744 \uacc4\uc18d \uc9c4\ud589\ud558\uac8c \ub420 \uc218 \uc788\ub2e4\ub294 \uac83\uc774\ub2e4. \uc774 \ubb38\uc81c\ub97c \ud574\uacb0\ud558\uae30 \uc704\ud574\uc11c \uac8c\uc784\uc758 \uc9c4\ud589 \uc0c1\ud669\uc774 \uc8fc\uc5b4\uc9c0\uba74 \uba87 \ubc88\uc9f8 \ucc28\ub840\uc5d0\uc11c \uc0ac\uc774\ud074\uc774 \uc644\uc131\ub418\uc5c8\ub294\uc9c0, \ud639\uc740 \uc544\uc9c1 \uac8c\uc784\uc774 \uc9c4\ud589 \uc911\uc778\uc9c0\ub97c \ud310\ub2e8\ud558\ub294 \ud504\ub85c\uadf8\ub7a8\uc744 \uc791\uc131\ud558\ub824 \ud55c\ub2e4.<\/p>\r\n\r\n<p>\uc785\ub825\uc73c\ub85c \uc810\uc758 \uac1c\uc218 <em>n<\/em>\uacfc <em>m<\/em> \ubc88\uc9f8 \ucc28\ub840\uae4c\uc9c0\uc758 \uac8c\uc784 \uc9c4\ud589 \uc0c1\ud669\uc774 \uc8fc\uc5b4\uc9c0\uba74 \uc0ac\uc774\ud074\uc774 \uc644\uc131 \ub418\uc5c8\ub294\uc9c0\ub97c \ud310\ub2e8\ud558\uace0, \uc644\uc131\ub418\uc5c8\ub2e4\uba74 \uba87 \ubc88\uc9f8 \ucc28\ub840\uc5d0\uc11c \ucc98\uc74c\uc73c\ub85c \uc0ac\uc774\ud074\uc774 \uc644\uc131\ub41c \uac83\uc778\uc9c0\ub97c \ucd9c\ub825\ud558\ub294 \ud504\ub85c\uadf8\ub7a8\uc744 \uc791\uc131\ud558\uc2dc\uc624.<\/p>\r\n","input":"<p>\uc785\ub825\uc740 \ud45c\uc900\uc785\ub825\uc744 \uc0ac\uc6a9\ud55c\ub2e4. \uc785\ub825\uc758 \uccab \ubc88\uc9f8 \uc904\uc5d0\ub294 \uc810\uc758 \uac1c\uc218\ub97c \ub098\ud0c0\ub0b4\ub294 \uc815\uc218 3 &le; <em>n<\/em> &le; 500,000 \uacfc \uc9c4\ud589\ub41c \ucc28\ub840\uc758 \uc218\ub97c \ub098\ud0c0\ub0b4\ub294 \uc815\uc218 3 &le; <em>m<\/em> &le; 1,000,000 \uc774 \uc8fc\uc5b4\uc9c4\ub2e4. \uac8c\uc784\uc5d0\uc11c \uc0ac\uc6a9\ud558\ub294 <em>n<\/em>\uac1c\uc758 \uc810\uc5d0\ub294 0 \ubd80\ud130 <em>n<\/em> &minus; 1 \uae4c\uc9c0 \uace0\uc720\ud55c \ubc88\ud638\uac00 \ubd80\uc5ec\ub418\uc5b4 \uc788\uc73c\uba70, \uc774 \uc911 \uc5b4\ub290 \uc138 \uc810\ub3c4 \uc77c\uc9c1\uc120 \uc704\uc5d0 \ub193\uc774\uc9c0&nbsp;\uc54a\ub294\ub2e4. \uc774\uc5b4\uc9c0\ub294 <em>m<\/em> \uac1c\uc758 \uc785\ub825 \uc904\uc5d0\ub294 \uac01\uac01 <em>i<\/em>\ubc88\uc9f8 \ucc28\ub840\uc5d0 \ud574\ub2f9 \ud50c\ub808\uc774\uc5b4\uac00 \uc120\ud0dd\ud55c \ub450 \uc810\uc758 \ubc88\ud638\uac00 \uc8fc\uc5b4\uc9c4\ub2e4 (1 &le; <em>i<\/em> &le; <em>m<\/em>).<\/p>\r\n","output":"<p>\ucd9c\ub825\uc740 \ud45c\uc900\ucd9c\ub825\uc744 \uc0ac\uc6a9\ud55c\ub2e4. \uc785\ub825\uc73c\ub85c \uc8fc\uc5b4\uc9c4 \ucf00\uc774\uc2a4\uc5d0 \ub300\ud574, <em>m<\/em> \ubc88\uc9f8 \ucc28\ub840\uae4c\uc9c0 \uac8c\uc784\uc744 \uc9c4\ud589\ud55c \uc0c1\ud669\uc5d0\uc11c \uc774\ubbf8 \uac8c\uc784\uc774 \uc885\ub8cc\ub418\uc5c8\ub2e4\uba74 \uc0ac\uc774\ud074\uc774 \ucc98\uc74c\uc73c\ub85c \ub9cc\ub4e4\uc5b4\uc9c4 \ucc28\ub840\uc758 \ubc88\ud638\ub97c \uc591\uc758 \uc815\uc218\ub85c \ucd9c\ub825\ud558\uace0, <em>m<\/em> \ubc88\uc758 \ucc28\ub840\ub97c \ubaa8\ub450 \ucc98\ub9ac\ud55c \uc774\ud6c4\uc5d0\ub3c4 \uc885\ub8cc\ub418\uc9c0 \uc54a\uc558\ub2e4\uba74 0\uc744 \ucd9c\ub825\ud55c\ub2e4.<\/p>\r\n","hint":"","original":"1","html_title":"0","problem_lang_tcode":"Korean"},{"problem_id":"20040","problem_lang":"1","title":"Cycle Game","description":"<p>Cycle game is a game in which two players take turns drawing a line segment for given points. The first player gets to draw a line segment in the odd-numbered turns, while the other player takes even-numbered turns. At the start of the game, <em>n<\/em> points are given in the plane, and each of the points has a unique number from 0 to <em>n<\/em> &minus; 1. Note that no three points lie on a same line. In each turn, a player draws a line segment connecting two given points. Players cannot redraw a line segment which has already been drawn, but it is allowed to draw a line segment across existing line segments. Players take turns until there is the loser who completes a cycle first. A cycle <em>C<\/em> is a subset of the line segments drawn by the players which satisfies the following condition.<\/p>\r\n\r\n<blockquote>\r\n<p>From any endpoint of a line segment in <em>C<\/em>, one can travel down and return to the end point while visiting every line segment in <em>C<\/em> exactly once.<\/p>\r\n<\/blockquote>\r\n\r\n<p>A problem is that as players draw many segments, it is difficult to check whether a cycle is created. Therefore, sometimes players continue to play the game even though a cycle has been made and the game is actually over. To prevent this situation, we need to write a program which examines whether the game has been over.<\/p>\r\n\r\n<p>Write a program to decide whether the game has been over and output in which turn the first cycle has been made.<\/p>\r\n","input":"<p>Your program is to read from standard input. The input starts with a line containing two integers 3 &le; <em>n<\/em> &le; 500,000 and 3 &le; <em>m<\/em> &le; 1,000,000, where <em>n<\/em> represents the number of points and <em>m<\/em> represents the number of turns in which line segments have been drawn. Every point has a unique number from 0 to <em>n<\/em> &minus; 1, and no three points lie on a same line. Each of the following <em>m<\/em> lines contains two numbers of points which are the end points of the line segments each player drew in the <em>i<\/em>-th turn (1 &le; <em>i<\/em> &le; <em>m<\/em>).<span style=\"display: none;\">&nbsp;<\/span><\/p>\r\n","output":"<p>Your program is to write to standard output. Print exactly one line. The line should contain the number of the turn in which the cycle has been first created, i.e., the game has been over, and 0 if the game was not over for <em>m<\/em> turns.<\/p>\r\n","hint":"","original":"0","html_title":"0","problem_lang_tcode":"English"}]

접근 방법

- 분리집합을 통해 부분집합들을 갱신해나가다가 유니온 파인드를 했을 때 같은 숫자가 나온 경우 몇번째 차례였는지 출력한다.

코드

# https://www.acmicpc.net/problem/20040
# 접근 방법
# 분리집합을 통해 부분집합들을 갱신해나가다가 유니온 파인드를 했을 때 같은 숫자가 나온 경우 몇번째 차례였는지 출력한다.
def get_parent(idx):
    if parent[idx] == idx:
        return idx
    index = get_parent(parent[idx])
    parent[idx] = index
    return index

def find_union(point1, point2):
    p1 = get_parent(point1)
    p2 = get_parent(point2)
    if p1 == p2:
        return True
    
    if p1 < p2:
        parent[p2] = p1
    else:
        parent[p1] = p2
    return False

import sys
input = sys.stdin.readline
n, m = map(int, input().split())
parent = [i for i in range(n+1)]
array = [list(map(int, input().split())) for _ in range(m)]
for i in range(1, m+1):
    point1, point2 = array[i-1]
    check = find_union(point1, point2)
    if check:
        break
    
if check:
    print(i)
else:
    print(0)
728x90
반응형