[2016-2017 ACM-ICPC, South Pacific Regional Contest (SPPC 16)]B Balloon Warehouse
[2016-2017 ACM-ICPC, South Pacific Regional Contest (SPPC 16)]C Crazy Rotations

[2016-2017 ACM-ICPC, South Pacific Regional Contest (SPPC 16)]J Just Terraffic

Zarxdy34 posted @ 2017年10月10日 19:29 in Codeforces with tags DP , 1099 阅读

    题目大意:将n个按升序排列的数字分割为一系列长度为2或3的小段,要求相邻且差<=1000的数必须在同一段,相邻且差>=2000的数不能在同一段,询问是否存在这样的分割方案,有则输出长度为2或3的小段的数量,多组解(分割方案不同但是两种小段的数量相同算同一种解)输出Ambiguous,无解输出Impossible。

    题解:比较显然可以看出来是动态规划,无解的处理很容易,但是多组解的判断比较困难。一种比较好的解决思路是,记f[i][0]和f[i][1]分别表示前i个数分割所称长度为2和3的小段的数量,更新时\[f[i][0]=max(f[i-2][0]+1,f[i-3][0])\]\[f[i][1]=max(f[i-2][1],f[i-3][1]+1)\],这样一来如果有多种解的话,\(2f[n][0]+3f[n][1]\)肯定是大于n的。

    无解的判断另开一个数组g记录即可。

#include <bits/stdc++.h>
using namespace std;
const int maxn=3e5+10;

long long f[maxn];
int g[maxn];
int t[maxn];
int n;

bool Check(int l,int r)
{
	for (int i=l;i<r;i++) if (t[i+1]-t[i]>=2000) return false;
	if (t[l]-t[l-1]<=1000) return false;
	if (t[r+1]-t[r]<=1000) return false;
	return true;
}

int main()
{
	scanf("%d",&n);
	for (int i=1;i<=n;i++) scanf("%d",&t[i]);
	t[0]=t[1]-2000;t[n+1]=t[n]+2000;
	g[0]=1;
	for (int i=2;i<=n;i++)
	{
		if (Check(i-1,i) && g[i-2]) f[i]=f[i-2]+1,g[i]=1;
		if (i>2 && Check(i-2,i) && g[i-3])
			f[i]=max(f[i]/n,f[i-3]/n+1)*n+max(f[i]%n,f[i-3]%n),g[i]=1;
	}
	if (g[n]==0) printf("Impossible\n");
	else if (2*(f[n]%n)+3*(f[n]/n)>n) printf("Ambiguous\n");
	else printf("Cars without trailers: %lld\nCars with trailers: %lld\n",f[n]%n,f[n]/n);
	return 0;
}

Avatar_small
DeaphetS 说:
2017年10月10日 19:31

前排膜zzy大佬


登录 *


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