Time does not change us. It just unfolds us.

Coding Test

[코드트리] 왕실의 기사 대결

소젬 2024. 4. 13. 23:03

https://www.codetree.ai/training-field/frequent-problems/problems/royal-knight-duel/description?page=1&pageSize=20

 

코드트리 | 코딩테스트 준비를 위한 알고리즘 정석

국가대표가 만든 코딩 공부의 가이드북 코딩 왕초보부터 꿈의 직장 코테 합격까지, 국가대표가 엄선한 커리큘럼으로 준비해보세요.

www.codetree.ai


왕실의 기사들은  크기의 체스판 위에서 대결을 준비하고 있습니다. 체스판의 왼쪽 상단은 (1,1)로 시작하며, 각 칸은 빈칸, 함정, 또는 벽으로 구성되어 있습니다. 체스판 밖도 벽으로 간주합니다.

왕실의 기사들은 자신의 마력으로 상대방을 밀쳐낼 수 있습니다. 각 기사의 초기위치는 로 주어지며, 그들은 방패를 들고 있기 때문에 를 좌측 상단으로 하며  크기의 직사각형 형태를 띄고 있습니다. 각 기사의 체력은 로 주어집니다.

(1) 기사 이동

왕에게 명령을 받은 기사는 상하좌우 중 하나로 한 칸 이동할 수 있습니다. 이때 만약 이동하려는 위치에 다른 기사가 있다면 그 기사도 함께 연쇄적으로 한 칸 밀려나게 됩니다. 그 옆에 또 기사가 있다면 연쇄적으로 한 칸씩 밀리게 됩니다. 하지만 만약 기사가 이동하려는 방향의 끝에 벽이 있다면 모든 기사는 이동할 수 없게 됩니다. 또, 체스판에서 사라진 기사에게 명령을 내리면 아무런 반응이 없게 됩니다.

(2) 대결 대미지

명령을 받은 기사가 다른 기사를 밀치게 되면, 밀려난 기사들은 피해를 입게 됩니다. 이때 각 기사들은 해당 기사가 이동한 곳에서  직사각형 내에 놓여 있는 함정의 수만큼만 피해를 입게 됩니다. 각 기사마다 피해를 받은 만큼 체력이 깎이게 되며, 현재 체력 이상의 대미지를 받을 경우 기사는 체스판에서 사라지게 됩니다. 단, 명령을 받은 기사는 피해를 입지 않으며, 기사들은 모두 밀린 이후에 대미지를 입게 됩니다. 밀렸더라도 밀쳐진 위치에 함정이 전혀 없다면 그 기사는 피해를 전혀 입지 않게 됨에 유의합니다.

 번에 걸쳐 왕의 명령이 주어졌을 때,  번의 대결이 모두 끝난 후 생존한 기사들이 총 받은 대미지의 합을 출력하는 프로그램을 작성해보세요.

입력 형식

첫 번째 줄에 , , 가 공백을 사이에 두고 주어집니다.

다음  개의 줄에 걸쳐서  크기의 체스판에 대한 정보가 주어집니다.

  • 이라면 빈칸을 의미합니다.
  • 이라면 함정을 의미합니다.
  • 라면 벽을 의미합니다.

다음  개의 줄에 걸쳐서 초기 기사들의 정보가 주어집니다. 이 정보는  순으로 주어지며, 이는 기사의 처음 위치는 를 좌측 상단 꼭지점으로 하며 세로 길이가 , 가로 길이가 인 직사각형 형태를 띄고 있으며 초기 체력이 라는 것을 의미합니다. 입력은 1번 기사부터 번 기사까지 순서대로 정보가 주어집니다.

단, 처음 주어지는 기사들의 위치는 기사들끼리 서로 겹쳐져 있지 않습니다. 또한 기사와 벽은 겹쳐서 주어지지 않습니다.

다음  개의 줄에 걸쳐 왕의 명령에 대한 정보가 주어집니다. 이 정보는  형태로 주어지며 이는 번 기사에게 방향 로 한 칸 이동하라는 명령임을 뜻합니다. 는 1 이상  이하의 값을 갖으며, 이미 사라진 기사 번호가 주어질 수도 있음에 유의합니다. 는 0, 1, 2, 3 중에 하나이며 각각 위쪽, 오른쪽, 아래쪽, 왼쪽 방향을 의미합니다.

  • : 체스판의 크기 ()
  • : 기사의 수 ()
  • : 명령의 수 ()
  • : 기사의 체력 ()

출력 형식

 개의 명령이 진행된 이후, 생존한 기사들이 총 받은 대미지의 합을 출력합니다.


#include <iostream>
#include <vector>
#include <algorithm>
#include <queue>
using namespace std;

int L, N, Q, ret = 0;
int _y[] = { -1,0,1,0 };
int _x[] = { 0,1,0,-1 };

bool cmp0(pair<int, int>& a, pair<int, int>& b)	// 위 : r 오름차순
{
	if (a.first == b.first) return a.second < b.second;
	return a.first < b.first;
}

bool cmp1(pair<int, int>& a, pair<int, int>& b)	// 오 : c 내림차순
{
	if (a.second == b.second) return a.first < b.first;
	return a.second > b.second;
}

bool cmp2(pair<int, int>& a, pair<int, int>& b)	// 아래 : r 내림차순
{
	if (a.first == b.first) return a.second < b.second;
	return a.first > b.first;
}

bool cmp3(pair<int, int>& a, pair<int, int>& b)	// 오 : c 내림차순
{
	if (a.second == b.second) return a.first < b.first;
	return a.second < b.second;
}

void printChess(vector<vector<int>> v)
{
	for (int i = 0; i < L; i++)
	{
		for (int j = 0; j < L; j++)
		{
			cout << v[i][j] << "	";
		}cout << endl;
	}cout << endl;
}


int main()
{
	cin >> L >> N >> Q;
	vector<vector<pair<int, int>>> knight;
	vector<vector<int>> chess(L, vector<int>(L, 0));
	vector<vector<int>> knightMap(L, vector<int>(L, -1));
	vector<vector<int>> newMap(L, vector<int>(L, 0));
	vector<int> firstHealth(N, 0);
	vector<int> health(N, 0);
	vector<pair<int, int>> command;
	
	for (int i = 0; i < L; i++)
	{
		for (int j = 0; j < L; j++)
		{
			cin >> chess[i][j];
		}
	}

	for (int i = 0; i < N; i++)
	{
		vector<pair<int, int>> tmp; 
		int r, c, h, w, k;
		cin >> r >> c >> h >> w >> k;
		
		for (int a = 0; a < h; a++)
		{
			for (int b = 0; b < w; b++)
			{
				int y = r - 1 + a, x = c - 1 + b;

				tmp.push_back(make_pair(y, x));

				knightMap[y][x] = i;
			}
		}
		knight.push_back(tmp);
		health[i] = k;
	}
	newMap = knightMap;
	firstHealth = health;

	for (int q = 0; q < Q; q++)
	{
		int i, d;
		cin >> i >> d;
		
		command.push_back(make_pair(i - 1, d));
	}

	//start
	for (pair<int, int> cmd : command)
	{
		vector<int> visit(N, 0);
		queue<int> q;
		vector<vector<pair<int, int>>> tmpkni(N);

		int i = cmd.first, d = cmd.second;
		if (!knight[i].size())continue;
		q.push(i);
		visit[i] = 1;

		bool endFlag = false;
		while (!q.empty() && !endFlag) {
			int mvKni = q.front();
			//sorting
			switch (d) {
			case 0:
			{
				sort(knight[mvKni].begin(), knight[mvKni].end(), cmp0);
				break;;
			}
			case 1:
			{
				sort(knight[mvKni].begin(), knight[mvKni].end(), cmp1);
				break;
			}
			case 2:
			{
				sort(knight[mvKni].begin(), knight[mvKni].end(), cmp2);
				break;
			}
			case 3:
			{
				sort(knight[mvKni].begin(), knight[mvKni].end(), cmp3);
				break;
			}
			}


			for (int n = 0; n < knight[mvKni].size(); n++)
			{
				if (endFlag) break;
				int sy = knight[mvKni][n].first, sx = knight[mvKni][n].second;
				int ny = sy + _y[d], nx = sx + _x[d];

				if (ny >= L || nx >= L || nx < 0 || ny < 0 || chess[ny][nx] == 2)	//walls
				{
					endFlag = true;
					break;
				}

				if (knightMap[ny][nx] != mvKni && knightMap[ny][nx] != -1)
				{
					if (visit[knightMap[ny][nx]] == 0)
					{
						q.push(knightMap[ny][nx]);	//미뤄질 기사
						visit[knightMap[ny][nx]] = 1;
					}
				}

				//save moving location
				if (newMap[sy][sx] == mvKni) newMap[sy][sx] = -1;
				newMap[ny][nx] = mvKni;
				tmpkni[mvKni].push_back(make_pair(ny, nx));
			}
			q.pop();
		}

		if (endFlag) //fail moving
		{
			newMap = knightMap;
			continue;
		}
		
		//success moving : calculate damage
		
		for (int idx = 0; idx < N; idx++)
		{
			if (tmpkni[idx].size() != 0)
				knight[idx] = tmpkni[idx];
		}

		for (int ki = 0; ki < N; ki++)
		{
			if (ki == i) continue; //command knight is exception 
			for (pair<int, int> kj : tmpkni[ki])
			{
				if (chess[kj.first][kj.second] == 1)
				{
					health[ki]--;
				}
			}
		}

		//die
		for (int h = 0; h < N; h++)
		{
			if (health[h] <= 0)
			{
				for (pair<int, int> kj : knight[h])
					newMap[kj.first][kj.second] = -1;
				knight[h].resize(0);
				health[h] = 0;
			}
		}

		knightMap = newMap;

	}

	for (int h = 0; h < N; h++) {
		if (health[h])
		{
			ret += firstHealth[h] - health[h];
		}
	}
	cout << ret << endl;
    return 0;
}