본문 바로가기

Algorithm

백준 15809번 전국시대

반응형

 

문제

전국시대엔 N개의 국가가 존재한다. 각 국가는 1부터 N까지의 번호를 가지고 있다.

또한, 모든 국가는 각자 자신의 국가의 힘을 상징하는 병력을 가지고 있다. 이때 M개의 기록이 주어진다. 각각의 기록은 다음과 같다.

  1. 동맹 - 두 나라가 서로 동맹을 맺는다. 두 나라의 병력이 하나로 합쳐진다.
  2. 전쟁 - 두 나라가 서로 전쟁을 벌인다. 병력이 더 많은 나라가 승리하며 패배한 나라는 속국이 된다. 이때 남은 병력은 승리한 나라의 병력에서 패배한 나라의 병력을 뺀 수치가 된다. 두 나라의 병력이 같을 경우 두 나라 모두 멸망한다.

모든 나라는 정직하기 때문에 내 동맹의 동맹도 나의 동맹이고, 내 동맹이 적과 전쟁을 시작하면 같이 참전한다. 속국인 경우도 동맹의 경우와 마찬가지이다.

따라서, 전쟁에서 진 국가와 동맹인 다른 국가 또한 전쟁에서 이긴 국가의 속국이 된다.

모든 기록이 끝났을 때 남아있는 국가의 수를 출력하고, 그 국가들의 남은 병력의 수를 오름차순으로 출력하는 프로그램을 작성하시오.

단, 여러 국가가 서로 동맹이거나 속국 관계인 경우는 한 국가로 취급한다.

입력

첫 번째 줄에 국가의 수를 나타내는 N과 기록의 수 M이 주어진다. (1 ≤ N, M ≤ 100,000)

두 번째 줄 부터 N개의 줄에 걸쳐 i번째 국가의 병력 Ai (1 ≤ i ≤ N)가 자연수로 주어진다. (1 ≤ Ai ≤ 10,000)

다음 M개의 줄에는 기록이 3개의 정수 O, P, Q로 주어진다. O가 1인 경우 P, Q가 서로 동맹을 맺었음을 의미하고, O가 2인 경우 P, Q가 서로 전쟁을 벌였음을 의미한다.

동맹끼리 다시 동맹을 맺거나 전쟁하는 입력은 주어지지 않는다.

출력

첫째 줄에 남아있는 국가의 수를 출력한다.

다음 줄에 각 국가의 남은 병력의 수를 띄어쓰기를 간격으로 오름차순으로 출력한다.

 

 

 

 

 

 

union find 응용문제 . 인덱스 시작 값 잘보고 배열 크기 잘 정할 것. 

 

 

 

 

#include <iostream>
#include  <algorithm>
#include  <vector>

using namespace std;
const int MAX = 100001;

vector <int> vec;

int N,M;



int size[MAX];
int parent[MAX];


int findparent(int Node)
{
if(Node == parent[Node])
return Node;


parent[Node] = findparent(parent[Node]);
return parent[Node];



}


void merge(int x, int y, int op)
{

int node1 = findparent(x);
int node2 = findparent(y);


if(node1 != node2)
{

 if(op ==1)
    {


 if (size[node1] < size[node2])
            {
    parent[node1] = node2;
    size[node2] += size[node1];
    size[node1] = 0;
            }
            
            else{
            
               parent[node2] = node1;
    size[node1] += size[node2];
    size[node2] = 0;
            
}
   
   }
    else
    {


 if (size[node1] < size[node2])
            {
    parent[node1] = node2;
    size[node2] -= size[node1];
    size[node1] = 0;
            }
            
            else{
            
               parent[node2] = node1;
    size[node1] -= size[node2];
    size[node2] = 0;
            
}
   
   }

}




}

















int main(void)
{

 ios_base::sync_with_stdio(0);

        cin.tie(0);
        

cin >> N >> M;

for(int i=1; i<=N; i++)
{
int a;

cin >> a;

size[i] = a;
parent[i] = i;


}




for(int i=1; i<=M; i++)
{
int o,p,q;

cin >> o >> p >> q;

   merge(p,q,o);


}

for(int i=1; i<=N; i++)
{


if(size[i] > 0)
{

vec.push_back(size[i]);
}

}



sort(vec.begin(),vec.end());

cout<<vec.size()<<endl;

for(int i=0; i<vec.size(); i++)
{
cout<<vec[i]<<" ";
}

cout<<endl;



}

반응형

'Algorithm' 카테고리의 다른 글

백준 11779번 최소비용 구하기2  (0) 2019.08.12
백준 16398번 행성연결  (0) 2019.08.11
백준 11286번 절댓값 힙  (0) 2019.08.09
백준 11724번 연결 요소의 개수  (0) 2019.08.08
백준 2804번 크로스워드 만들기  (0) 2019.08.07