HDU 4293 Groups(区间dp)

HDU 4293

题意:有 n 个人,可任意分成若干组,然后每个人各提供一个信息,表示他们组前面有多少个人,后面有多少个人。问最多有多少个信息是真实的的。

思路:

这道题一开始给我的印象是什么乱七八糟的东西,后来也没想通到底该怎么做,好在赛后百度在手天下我有:)

我们可以把 这n个人看成一段区间 [1,n]。

设每个人的信息是a、b,则这个信息代表了他们组所在的区间 [a+1,n-b]。

若a+b>n,显然撒谎,跳过不做处理。

我们用一个Map[i][j]数组将同在一区间[i,j]的人数纪录下来,纪录过程中若Map[i][j] > n-i-j则说明提供[i,j]区间消息的人里有人撒谎,略过不计,令Map[i][j] = n-i-j;

问题转化成了在 [1,n] 这段区间中分布的若干个带有权值的区间,问如何选取一些不相交的区间,使权值之和最大。

我们用dp[i]表示[1,i]区间权值之和的最大值,转移方程为dp[i] = max(dp[i],dp[j]+map[j][i]).

code:

/*
* @author Novicer
* language : C++/C
*/
#include<iostream>
#include<sstream>
#include<fstream>
#include<vector>
#include<list>
#include<deque>
#include<queue>
#include<stack>
#include<map>
#include<set>
#include<bitset>
#include<algorithm>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cctype>
#include<cmath>
#include<ctime>
#include<iomanip>
#define INF 2147483647
#define cls(x) memset(x,0,sizeof(x))
#define rise(i,a,b) for(int i = a ; i <= b ; i++)
using namespace std;
const double eps(1e-8);
typedef long long lint;

const int maxn = 500 + 5;

int Map[maxn][maxn];
int dp[maxn];

int main(){
//	freopen("input.txt","r",stdin);
	int n;
	while(cin >> n){
		cls(Map);
		for(int i = 1 ; i <= n ; i++){
			int a,b;
			scanf("%d%d",&a,&b);
			if(a+b >= n)
				continue;
			else{
				if(n-a-b > Map[a+1][n-b])
					Map[a+1][n-b]++;
				else
					Map[a+1][n-b] = n-a-b;
			}
		}
		cls(dp);
		for(int i = 1 ; i <= n ; i++)
			for(int j = 1 ; j <= i ; j++)
				dp[i] = max(dp[i] , dp[j-1]+Map[j][i]);
		cout << dp[n] << endl;
	}

	return 0;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-12 19:17:50

HDU 4293 Groups(区间dp)的相关文章

HDU 4293 Groups (线性dp)

OJ题目:click here~~ 题目分析:n个人分为若干组 , 每一个人描写叙述其所在的组前面的人数和后面的人数.求这n个描写叙述中,最多正确的个数. 设dp[ i ] 为前i个人的描写叙述中最多正确的个数,则dp[ n ] 为要求的.num[ i ][ j ]  保存说前面有i个人 , 后面有j个人的人数,显然num[ i ][ j ]不超过n - i - j; 转移方程dp[ i ] = max(dp[ i ] , dp[ j ]  + num[ j ][ n - i ])  ,详解见代

HDU - 4293 Groups (DP)

题目大意:有N个人,人人之间可以组成一个团队,现在N个人各说一句话,说自己前面有多少人,后面有多少人 现在要求你判断这N个人中最多有多少人说真话 解题思路:参考了别人的 设有n个人,其中有一个人说了他前面有a个人,后面有b个人,那么他所在的区间就变成了[a + 1, n - b],那么就可以将这个人归到[a + 1, n - b] 如果[a + 1, n - b]的区间的人数超过了 n - a -b,那么就可以将其他的人忽略掉,因为这个区间最多有n-a-b个人 那么现在的问题就变成了,如何选择不

Hdu 2513 区间DP

Cake slicing Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 149    Accepted Submission(s): 86 Problem Description A rectangular cake with a grid of m*n unit squares on its top needs to be slice

HDU 5900(区间DP)

HDU 5900 QSC and Master 题意:给一串数的key和value,如果相邻两元素key不是互质的就可以将这俩移除并获得这俩的value值,移除后两侧的元素便是相邻了,问最终最大能获得多少value值. 思路:区间DP,区间长度为1时dp[i][i]为0,添加一个标记数组vis[i][j],记录区间内是否所有数字都为可以移除,每次处理需要讨论最后留下区间两侧的情况. #include <cstdio> #include <cstring> #include <

hdu 5396 Expression(区间dp)

Problem Description Teacher Mai has n numbers a1,a2,?,anand n−1 operators("+", "-" or "*")op1,op2,?,opn−1, which are arranged in the form a1 op1 a2 op2 a3 ? an.He wants to erase numbers one by one. In i-th round, there are n+

HDU 5115 (区间DP)

题目大意:你是一个战士现在面对,一群狼,每只狼都有一定的主动攻击力和附带攻击力.你杀死一只狼.你会受到这只狼的(主动攻击力+旁边两只狼的附带攻击力)这么多伤害~现在问你如何选择杀狼的顺序使的杀完所有狼时,自己受到的伤害最小.(提醒,狼杀死后就消失,身边原本相隔的两只狼会变成相邻,而且不需要考虑狼围城环这种情况) 输入要求:总共T组数据,每组N只狼,按顺序输入全部狼的主动攻击力和然后再按顺序输入全部狼的附带攻击力 输出要求:Case #x: y x第几组数据,y最少受到的伤害. 思路:枚举当前区间

HDU 4293 Groups

Groups Time Limit: 1000ms Memory Limit: 32768KB This problem will be judged on HDU. Original ID: 429364-bit integer IO format: %I64d      Java class name: Main After the regional contest, all the ACMers are walking alone a very long avenue to the din

HDU 5568 sequence2 区间dp+大数

题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5568 题意: 求所有长度为k的严格升序子序列的个数. 题解: 令dp[i][k]表示以i结尾的长度为k的所有严格升序子序列的个数,则有状态转移:dp[i][k]+=dp[j][k-1](其中,arr[i]>arr[j],并且i>j): 最后答案为dp[1][k]+...dp[n][k]. 代码: 1 import java.util.*; 2 import java.math.*; 3 publ

You Are the One HDU - 4283(区间dp)

You Are the One Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 4762    Accepted Submission(s): 2264 Problem Description The TV shows such as You Are the One has been very popular. In order to