hdu 4597 Play Game(記憶化搜索)

題目連接:hdu 4597 Play Gamephp


題目大意:給出兩堆牌,僅僅能從最上和最下取,而後兩我的輪流取,都依照本身最優的策略。問說第一我的對多的分值。spa


解題思路:記憶化搜索,狀態出來就很水,dp[fl][fr][sl][sr][flag],表示第一堆牌上邊取到fl,如下取到fr,相同sl。sr爲第二堆牌,flag爲第幾我的在取。假設是第一我的,dp既要儘可能大,假設是第二我的,那麼確定儘可能小。code


#include <cstdio>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 25;
const int INF = 0x3f3f3f3f;
int n, f[N], s[N], dp[N][N][N][N][2];

void init () {
	memset(dp, -1, sizeof(dp));

	scanf("%d", &n);
	for (int i = 1; i <= n; i++)
		scanf("%d", &f[i]);
	for (int i = 1; i <= n; i++)
		scanf("%d", &s[i]);
}

int solve (int fl, int fr, int sl, int sr, int flag) {
	int& ans = dp[fl][fr][sl][sr][flag];
	if (fl > fr && sl > sr)
		return ans = 0;

	if (ans != -1)
		return ans;

	if (flag) {
		ans = 0;
		if (fl <= fr) {
			ans = max(ans, solve(fl+1, fr, sl, sr, 1-flag) + f[fl]);
			ans = max(ans, solve(fl, fr-1, sl, sr, 1-flag) + f[fr]);
		}

		if (sl <= sr) {
			ans = max(ans, solve(fl, fr, sl+1, sr, 1-flag) + s[sl]);
			ans = max(ans, solve(fl, fr, sl, sr-1, 1-flag) + s[sr]);
		}
	} else {
		ans = INF;
		if (fl <= fr) {
			ans = min(ans, solve(fl+1, fr, sl, sr, 1-flag));
			ans = min(ans, solve(fl, fr-1, sl, sr, 1-flag));
		}

		if (sl <= sr) {
			ans = min(ans, solve(fl, fr, sl+1, sr, 1-flag));
			ans = min(ans, solve(fl, fr, sl, sr-1, 1-flag));
		}
	}
	return ans;
}

int main () {
	int cas;
	scanf("%d", &cas);
	while (cas--) {
		init ();
		printf("%d\n", solve(1, n, 1, n, 1));
	}
	return 0;
}
相關文章
相關標籤/搜索