인프런 커뮤니티 질문&답변

요가인님의 프로필 이미지
요가인

작성한 질문수

10주완성 C++ 코딩테스트 | 알고리즘 코딩테스트

3-O

3-O 질문있습니다

작성

·

240

0

왜 틀렸는지 도저히 모르겠습니다..

 

#include <iostream>
#include <cstring>
using namespace std;

const int dy[2] = { 0,0 };
const int dx[2] = { -1, 1 };

int N, M, H, a, b, adj[31][21], visited[31][21], ret[21], cnt;
bool isOk;

void Print()
{
	cout << endl;
	for (int y = 0; y < 31; y++)
	{
		for (int x = 0; x < 21; x++)
			cout << adj[y][x];
		cout << endl;
	}
	cout << endl;
}

void Move(int y, int x, int start)
{
	if (y == H + 1)
	{
		ret[start] = x;
		return;
	}

	bool isDown = true;

	visited[y][x] = start;

	// 좌우 확인
	for (int i = 0; i < 2; i++)
	{
		int ny = y;
		int nx = x + dx[i];

		if (ny > H + 1 || nx <= 0 || nx > 2 * N - 1)
			continue;
		if (visited[ny][nx] == start)
			continue;
		if (adj[ny][nx] == 0)
			continue;

		Move(ny, nx, start);
		isDown = false;
	}

	if (isDown)
		Move(y + 1, x, start);
}

bool Check()
{
	// 값들 초기화
	memset(ret, 0, sizeof(ret));
	memset(visited, 0, sizeof(visited));

	for (int x = 1; x <= N; x++)
	{
		Move(0, x * 2 - 1, x * 2 - 1);
		if (ret[x * 2 - 1] != x * 2 - 1)
		{
			return false;
		}
	}

	return true;
}

void AddLadder()
{
	if (isOk)
		return;

	if (cnt >= 3)
		return;

	for (int y = 1; y <= H; y++)
		for (int x = 1; x < N; x++)
		{
			if (adj[y][x * 2] != 1 && adj[y][x * 2 - 2] != 1 && adj[y][x * 2 + 2] != 1)
			{
				adj[y][x * 2] = 1;
				cnt++;

				if (Check())
				{
					isOk = true;
					// Print();
					return;
				}
				AddLadder();

				adj[y][x * 2] = 0;
				cnt--;
			}
		}

	return;
}


int main()
{
	cin >> N >> M >> H;

	// 사다리 초기값
	for (int y = 0; y <= H + 1; y++)
		for (int x = 1; x <= N; x++)
			adj[y][x * 2 - 1] = 1;

	// 가로선 정보
	for (int i = 0; i < M; i++)
	{
		cin >> a >> b;
		adj[a][b * 2] = 1;
	}

	if (Check())
		cout << 0;
	else
	{
		AddLadder();
		if (cnt == 0)
			cout << -1;
		else
			cout << cnt;
	}

	return 0;
}

답변 1

0

큰돌님의 프로필 이미지
큰돌
지식공유자

안녕하세요 가인님 ㅎㅎ

코드 나름 잘 짜셨는데요 ㅎㅎ

다만 코드가 잘못된 부분이 2가지가 있습니다.

 

먼저 해당 좌표마다 무조건 1부터 탐색을 하게 되면 너무나 비효율적입니다. 좌표를 매개변수로 넘기고 해당 매개변수부터 시작하게 바꿔주세요.

	for (int y = 1; y <= H; y++)
		for (int x = 1; x < N; x++)
		{
			if (adj[y][x * 

 

그리고 이부분인데요.

			if (adj[y][x * 2] != 1 && adj[y][x * 2 - 2] != 1 && adj[y][x * 2 + 2] != 1)
			{
				adj[y][x * 2] = 1;
				cnt++;

				if (Check())
				{
					isOk = true; 
					return;
				}

그러나 check() 가 먼저 되는 순간 무조건 그것이 글로벌한 최적해다. 그렇게 해서 빠져나온다고 되어있는데

그렇게 치면 예를 들어 이렇게 빨리 찾아서 3을 내뱉고 끝내겠죠.

|-|-|-|-|-|-|-|-|-|-|-|-|-|-|-|

|-|-|-|-|-|-|-|-|-|-|-|-|-|-|-|

|-|-|-|-|-|-|-|-|-|-|-|-|-|-|-|

 

하지만

|-|-|-|-|-|-|-|-|-|-|-|-|-|-|-|

|-|-|-|-|-|-|-|-|-|-|-|-|-|-|-|

|-|-|-|-|-|-|-|-|-|-|-|-|-|-|-|

이런 경우가 최적해인 경우는 알아차리지 못하지 않을까요? 즉 모든 경우의 수를 판단하지 못합니다.

이런 부분을 고쳐주시고 cnt를 아주큰 값에서 min으로 갱신될 수 있도록 바꿔주셔야 합니다.

최솟값은 최댓값부터 갱신시켜주시면서 만들어주셔야 해요.



또 질문 있으시면 언제든지 질문 부탁드립니다.

좋은 수강평과 별점 5점은 제게 큰 힘이 됩니다. :)

감사합니다.

강사 큰돌 올림.

 

요가인님의 프로필 이미지
요가인

작성한 질문수

질문하기