1691: [Usaco2007 Dec]挑剔的美食家
2600: [Ioi2011]ricehub

2298: [HAOI2011]problem a

高橋直樹 posted @ 2013年4月01日 15:51 in BZOJ with tags HAOI bzoj , 1340 阅读

http://www.lydsy.com/JudgeOnline/problem.php?id=2298

RunID User Problem Result Memory Time Language Code_Length Submit_Time
381428 Liu_Ts 2298 Accepted 2740 kb 444 ms C++ 797 B 2013-04-01 15:23:28

 

题意

n个人,每个人说Xi个人比我低,Yi个人比我高

问最少有多少个人说假话

题解

其实想的和题解一样,但是忽略题解中所说的情况,一直wa,

掉(xi + yi ≥ n)的,

每句话都可以表示成一个[yi + 1, n - xi]的区间。

一开始所有的区间权值为1;

然后把同样的区间合并,权值设为合并在一起的区间数量

选出最多的互补相交的区间,记为k

那么答案就是n-k。

需要注意的是如果某个同样区间的数量>区间长度,要改成区间长度

 

#include<cstdio>
#include<algorithm>
#include<vector>
#define MP make_pair
#define PB push_back
#define s first
#define t second
using namespace std;
typedef pair<int,int> pr;
int n,f[100001],tot;
vector<pr> p;

bool cmp(pr a,pr b)
{
	if(a.t!=b.t) return a.t<b.t;
	else return a.s<b.s;
}
int main()
{
	scanf("%d",&n);
	for(int i=0;i<n;i++)
	{
		int x,y;scanf("%d%d",&x,&y);
		if(x+y>=n)	continue;
		p.PB(MP(y+1,n-x));tot++;
	}
	sort(p.begin(),p.end(),cmp);
	int j=0;
	for(int i=1;i<=n;i++)
	{
		f[i]=f[i-1];
		while(j<tot && p[j].t==i)
		{
			int k;for(k=j;k<tot && p[k]==p[j];k++);
			f[i]=max(f[i],f[p[j].s]+min(k-j,p[j].t-p[j].s));
			j=k;
		}
	}
	printf("%d\n",n-f[n]);
}

 


登录 *


loading captcha image...
(输入验证码)
or Ctrl+Enter