首页 > 其他分享 >P1541 [NOIP2010 提高组] 乌龟棋题解

P1541 [NOIP2010 提高组] 乌龟棋题解

时间:2024-02-17 21:00:58浏览次数:17  
标签:NOIP2010 int 题解 scanf 45 ++ step mp P1541

有两种方法,代码注释都很详细了直接上代码

一:记忆化搜索

#include<bits/stdc++.h>
using namespace std;
int t[15];
int n, m;
int a[400];
int mp[45][45][45][45];//mp[i][j][k][l]表示1号用i张,2号用j张,3号用k张,4号用l张的情况下,最多能拿多少分 
int dfs(int step, int w)//step表示处理了几张卡片   w是现在的位置 
{
	if (mp[t[1]][t[2]][t[3]][t[4]])//如果已经有值 
		return mp[t[1]][t[2]][t[3]][t[4]];//直接返回 
	if (step == m)//都决定完了 
		return a[w];//返回此位置的值 
	int cnt = 0;
	for (int i = 1;i <= 4; i++)
		if (t[i] != 0)
		{
			t[i]--;
			cnt = max(dfs(step + 1, w + i), cnt);//找后面所有情况中的最大值 
			t[i]++;
		}
	return mp[t[1]][t[2]][t[3]][t[4]] = cnt + a[w];//加上此位置的值为该状态下的最大值 
}
int main()
{
	scanf ("%d%d", &n, &m);
	for (int i = 0;i < n; i++)
		scanf ("%d", a + i);
	for (int i = 0, x;i < m; i++)
	{
		scanf ("%d", &x);
		t[x]++;
	}
	dfs(0, 0);
	printf ("%d", mp[t[1]][t[2]][t[3]][t[4]]);
	return 0;
}

二:用神奇的四重循环

#include<bits/stdc++.h>
using namespace std;
const int N = 45;
int n, m;
int f[N][N][N][N];//f[i][j][k][l]表示1号用i张,2号用j张,3号用k张,4号用l张时最多能拿多少分 
int t[10];
int a[355]; 
int main()
{
	scanf ("%d%d", &n, &m);
	for (int i = 0;i < n; i++)
		scanf ("%d", a + i);
	for (int i = 0, x;i < m; i++)
	{
		scanf ("%d", &x);
		t[x]++;
	}
	f[0][0][0][0] = a[0];//初始化 
	for (int t1 = 0;t1 <= t[1]; t1++)//循环1的张数 
		for (int t2 = 0;t2 <= t[2]; t2++)//循环2的张数 
			for (int t3 = 0;t3 <= t[3]; t3++)//循环3的张数 
				for (int t4 = 0;t4 <= t[4]; t4++)//循环4的张数 
				{
					int tt = t1 + t2 * 2 + t3 * 3 + t4 * 4;//算出现在的位置 (我是从0开始的) 
					if (t1)//如果1的张数不唯一,就可以求出去1张1,并求最大值 
						f[t1][t2][t3][t4] = max(f[t1][t2][t3][t4], f[t1 - 1][t2][t3][t4] + a[tt]);
					if (t2)//同上 
						f[t1][t2][t3][t4] = max(f[t1][t2][t3][t4], f[t1][t2 - 1][t3][t4] + a[tt]);
					if (t3)//同上 
						f[t1][t2][t3][t4] = max(f[t1][t2][t3][t4], f[t1][t2][t3 - 1][t4] + a[tt]);
					if (t4)//同上 
						f[t1][t2][t3][t4] = max(f[t1][t2][t3][t4], f[t1][t2][t3][t4 - 1] + a[tt]);
				}
	//结果在用 1号用t[1]张,2号用t[2]张,3号用t[3]张,4号用t[4]张里
	//即在 f[t[1]][t[2]][t[3]][t[4]]里 
	printf ("%d", f[t[1]][t[2]][t[3]][t[4]]);
	return 0;
}

标签:NOIP2010,int,题解,scanf,45,++,step,mp,P1541
From: https://www.cnblogs.com/Assassins-Creed/p/18018400

相关文章

  • TopCoder SRM478C RandomApple 题解
    题意:有\(k\)种苹果和\(n\)个箱子,每个箱子中有一些苹果,先等概率选取\(n\)个箱子组成集合的非空子集,再从选出的苹果中随机选一个,问每种苹果被选中的概率是多少箱子\(i\)有\(a_{i,j}\)个第\(j\)种苹果,第\(i\)个箱子的总苹果数\(siz_i=\sum\limits_{j=1}^ka_{i,j}\),苹果总数\(sum=\su......
  • 无限酒店 题解
    题目链接由于间隔不变,对于下面\(3\)个操作,只需记录起始位置与间隔即可。对于无数人到达酒店:所有位置的起始点\(\times2\),间隔\(\times2\),新的团队起始点为\(1\),间隔为\(2\)。对于\(k\)个人到达酒店:所有点的起始点\(+k\),间隔不变,新的团队起始点为\(0\),间隔为\(1\)......
  • CF1365G Secure Password 题解
    Description本题是交互题。有一个固定的数组\(A\),同时通过数组\(A\)构造出数组\(P\),具体来讲,\(P_i\)是\(A\)中除\(A_i\)外的所有元素的按位或。你需要在最多\(13\)次询问中得到最后的\(P\)数组。\(2\leqn\leq1000\)。Solution首先有一个\(2\logn\)的是注......
  • 传纸条 题解
    题目描述小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了。幸运的是,他们可以通过传纸条来进行交流。纸条要经由许多同学传到对方手里,小渊坐在矩阵......
  • 2024.2.17模拟赛T1题解
    先考虑\(q=(1...n)\)的情况:发现如果设\(divcnt(p)\)表示将\(p\)划分为极小值域连续段的个数,那满足\(divcnt(p)\gem\)的排列都是合法的。那现在要求出有多少个排列符合条件可以先算出长度为\(i\),\(divnct\)为\(1\)的排列个数,这可以用dp解决然后再背包一下,就能求......
  • P2042 [NOI2005] 维护数列 题解
    题目链接:维护数列比较不好码的题,首先确保自己会一种文艺平衡树的书写,这点因人而异,比较推荐初学者学\(fhq\)平衡树,坑比较少,比较好码,写平衡树合并、持久化类的题时,也比较好写。注意到空间需求比较大,还涉及删除,我们的删除用各种树类数据结构中最常用的回收标记用于新的节点开辟。......
  • CF1929E 题解
    题意:给定一棵\(n\)个节点数和\(k\)条路径\((a_i,b_i)\),求至少将多少条边染色,使得给定路径都至少有一条染色的边。\(n\le10^5,k\le20\)。思路:好题。显然状压\(dp\),\(dp[S]\)表示至少染多少条边使得\(S\)中的路径都满足条件。正常思路是枚举子集转移,考虑\(T\s......
  • 整数划分 题解
    题目描述如何把一个正整数N(N长度<20)划分为M(M>1)个部分,使这M个部分的乘积最大。N、M从键盘输入,输出最大值及一种划分方式。输入格式第一行一个正整数T(T<=10000),表示有T组数据。接下来T行每行两个正整数N,M。输出格式对于每组数据第一行输出最大值。第二行输出划分方案,将N按......
  • 情书密码 题解
    题目描述有消息称:绝恋找到了自己的MissRight,正准备自己的表白。绝恋已经写好了情书,但为了避免其它人截获,他对情书进行加密。为了打探绝恋的私密,你冒着生命危险终于搞到了这封情书。原以为可以轻易将情书解密,结果竟然发现聪明的绝恋并没有直接写出加密用的密码,而是在那粉红色的......
  • CF484B题解
    朴素的办法是枚举每两个数然后更新取模的结果。时间复杂度为\(O(n^2)\)不能通过。这个朴素的过程可以看作枚举一个数然后找对其取模最大值的过程。我们可以枚举一个数,然后再枚举以它的倍数为两端的区间,找其中取模的最大值。找最大值的过程可以二分或双指针。值域很小,也可以......