結果

問題 No.421 しろくろチョコレート
ユーザー asugen0402
提出日時 2019-04-05 15:47:18
言語 C
(gcc 13.3.0)
結果
WA  
実行時間 -
コード長 5,247 bytes
コンパイル時間 377 ms
コンパイル使用メモリ 34,304 KB
実行使用メモリ 6,948 KB
最終ジャッジ日時 2024-06-11 17:53:48
合計ジャッジ時間 2,186 ms
ジャッジサーバーID
(参考情報)
judge4 / judge5
このコードへのチャレンジ
(要ログイン)
ファイルパターン 結果
other AC * 52 WA * 13
権限があれば一括ダウンロードができます

ソースコード

diff #

#include <float.h>
#include <limits.h>
#include <math.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>

// 内部定数
#define D_SIZE_MAX		55										// 最大サイズ

// 内部変数
static FILE *szpFpI;											// 入力
static int siW, siH;											// 幅・高さ
static char sc2Cho[D_SIZE_MAX][D_SIZE_MAX];						// チョコ
static int si2Cnt[D_SIZE_MAX][D_SIZE_MAX];						// 隣接チョコ数
static int siSum;												// 幸福度

// 内部変数 - テスト用
#ifdef D_TEST
	static int siRes;
	static FILE *szpFpA;
	static int siTNo;
#endif

// 出力
int
fOut(
	char *pcpLine				// <I> 1行
)
{
	char lc1Buf[1024];

#ifdef D_TEST
	fgets(lc1Buf, sizeof(lc1Buf), szpFpA);
	if (strcmp(lc1Buf, pcpLine)) {
		siRes = -1;
	}
#else
	printf("%s", pcpLine);
#endif

	return 0;
}

// チョコ判定
int
fIsCho(
	int piX						// <I> 位置X
	, int piY					// <I> 位置Y
)
{
	if (sc2Cho[piY][piX] == 'w') {
		return 1;
	}
	if (sc2Cho[piY][piX] == 'b') {
		return 1;
	}

	return 0;
}

// 隣接チョコ数 - 加算
int
fAddCnt(
	int piX						// <I> 位置X
	, int piY					// <I> 位置Y
	, int piAdd					// <I> 加算値
)
{
	si2Cnt[piY + 1][piX] += piAdd;
	si2Cnt[piY - 1][piX] += piAdd;
	si2Cnt[piY][piX + 1] += piAdd;
	si2Cnt[piY][piX - 1] += piAdd;

	return 0;
}

// 隣接チョコ数 - 取得
int
fGetCnt(
	int piX						// <I> 位置X
	, int piY					// <I> 位置Y
)
{
	int liRet;

	// チョコ判定
	liRet = fIsCho(piX, piY);
	if (liRet == 0) {
		return INT_MAX;
	}

	return si2Cnt[piY][piX];
}

// 4方向の隣接チョコ数最小マス - 取得
int
fGetCnt4(
	int piX						// <I> 位置X
	, int piY					// <I> 位置Y
	, int *pipX					// <O> 隣接位置X
	, int *pipY					// <O> 隣接位置Y
)
{
	// 右
	int liMin = fGetCnt(piX + 1, piY);
	*pipX = piX + 1;
	*pipY = piY;

	// 左
	int liCnt = fGetCnt(piX - 1, piY);
	if (liMin > liCnt) {
		liMin = liCnt;
		*pipX = piX - 1;
		*pipY = piY;
	}

	// 下
	liCnt = fGetCnt(piX, piY + 1);
	if (liMin > liCnt) {
		liMin = liCnt;
		*pipX = piX;
		*pipY = piY + 1;
	}

	// 上
	liCnt = fGetCnt(piX, piY - 1);
	if (liMin > liCnt) {
		liMin = liCnt;
		*pipX = piX;
		*pipY = piY - 1;
	}

	return liMin;
}

// 方法3
int
fProc3(
)
{
	int i, j, liRet;

	// 隣接チョコ数 - セット
	for (i = 1; i <= siH; i++) {
		for (j = 1; j <= siW; j++) {
			liRet = fIsCho(j, i);
			fAddCnt(j, i, liRet);
		}
	}

	while (1) {

		// 隣接チョコ数最小マス - 取得
		int liMin1 = 5;
		int liX1, liY1, liMin2, liX2, liY2;
		for (i = 1; i <= siH; i++) {
			for (j = 1; j <= siW; j++) {

				// 隣接チョコ数
				int liCnt = fGetCnt(j, i);
				if (liMin1 < liCnt) {
					continue;
				}

				// 4方向の隣接チョコ数最小マス
				int liX, liY;
				int liCnt2 = fGetCnt4(j, i, &liX, &liY);
				if (liMin1 > liCnt) {
					if (liCnt2 == INT_MAX) {
						continue;
					}
				}
				else  {
					if (liMin2 <= liCnt2) {
						continue;
					}
				}

				// 更新
				liMin1 = liCnt;
				liX1 = j;
				liY1 = i;
				liMin2 = liCnt2;
				liX2 = liX;
				liY2 = liY;
			}
		}
		if (liMin1 == 5) {
			break;
		}

		// 食べる
		siSum += 100;
		sc2Cho[liY1][liX1] = 0;
		sc2Cho[liY2][liX2] = 0;
		fAddCnt(liX1, liY1, -1);
		fAddCnt(liX2, liY2, -1);
	}

	return 0;
}

// 方法2・1
int
fProc21(
)
{
	int i, j;

	// チョコ数 - 取得
	int liWCnt = 0;
	int liBCnt = 0;
	for (i = 1; i <= siH; i++) {
		for (j = 1; j <= siW; j++) {
			if (sc2Cho[i][j] == 'w') {
				liWCnt++;
			}
			else if (sc2Cho[i][j] == 'b') {
				liBCnt++;
			}
		}
	}

	// 幸福度
	if (liWCnt > liBCnt) {
		siSum += liBCnt * 10;
		siSum += liWCnt - liBCnt;
	}
	else {
		siSum += liWCnt * 10;
		siSum += liBCnt - liWCnt;
	}

	return 0;
}

// 実行メイン
int
fMain(
)
{
	int i;
	char lc1Buf[1024];

	// 幅・高さ - 取得
	fgets(lc1Buf, sizeof(lc1Buf), szpFpI);
	sscanf(lc1Buf, "%d%d", &siH, &siW);

	// チョコ - 取得
	for (i = 1; i <= siH; i++) {
		fgets(&sc2Cho[i][1], sizeof(sc2Cho[0]) - 1, szpFpI);
	}

	// 方法3
	fProc3();

	// 方法2・1
	fProc21();

	return 0;
}

// 1回実行
int
fOne(
)
{
	int liRet;
	char lc1Buf[1024];

	// データ - 初期化
	memset(sc2Cho, 0, sizeof(sc2Cho));							// チョコ
	memset(si2Cnt, 0, sizeof(si2Cnt));							// 隣接チョコ数
	siSum = 0;													// 幸福度

	// 入力 - セット
#ifdef D_TEST
	sprintf(lc1Buf, ".\\Test\\T%d.txt", siTNo);
	szpFpI = fopen(lc1Buf, "r");
	sprintf(lc1Buf, ".\\Test\\A%d.txt", siTNo);
	szpFpA = fopen(lc1Buf, "r");
	siRes = 0;
#else
	szpFpI = stdin;
#endif

	// 実行メイン
	liRet = fMain();

	// 結果 - セット
	sprintf(lc1Buf, "%d\n", siSum);

	// 結果 - 出力
	fOut(lc1Buf);

	// 残データ有無
#ifdef D_TEST
	lc1Buf[0] = '\0';
	fgets(lc1Buf, sizeof(lc1Buf), szpFpA);
	if (strcmp(lc1Buf, "")) {
		siRes = -1;
	}
#endif

	// テストファイルクローズ
#ifdef D_TEST
	fclose(szpFpI);
	fclose(szpFpA);
#endif

	// テスト結果
#ifdef D_TEST
	if (siRes == 0) {
		printf("OK %d\n", siTNo);
	}
	else {
		printf("NG %d\n", siTNo);
	}
#endif

	return 0;
}

// プログラム開始
int
main()
{

#ifdef D_TEST
	int i;
	for (i = D_TEST_SNO; i <= D_TEST_ENO; i++) {
		siTNo = i;
		fOne();
	}
#else
	fOne();
#endif

	return 0;
}

0