BZOJ 2708 [Violet 1]木偶 DP

题意:链接

方法: DP

解析:

这题太神辣。

做梦都没想到DP啊,反正我不会。

先谈一个我有过的错的想法。

最小费用最大流?

能匹配的边连费用为1的,不能匹配的连费用为0的

跑最小费用最大流

然而这显然是错的,我还思考半天。

因为这道题强制如果还用费用为1的边,那必须先跑费用为1的边。

这样就不符合辣

至于自己改下这个写法?

尝试过- -,然而卡在哪里呢?

一堆费用为1的边你先跑那个呢?- -评估?评估函数怎么写啊,我不会啊。

所以这个写法显然弃疗。

然而我还曾有过一个想法。

贪心?

对于每一个值pi来说,我们统计一下Num,然后呢

我们来考虑拿最大的Num,最大的Num怎么拿呢?

让他没有匹配的对象,或是尽可能的使他能匹配的对象都跟别人匹配,然后Num-他还能匹配的对象就是这一轮操作的答案。

每匹配一次更新一遍,直到Num都为0

不过这正确性有点可怜…

总感觉是错的。

好像上一个方法的错误部分就能卡住这个?

应该吧,反正我没实现。

但总感觉贪心有路子,但是我不会啊。

正解

正解好神而且好简洁啊。

排一下序

然后设f[i]为前i个最多扔多少个。

f[i]=max(f[j],cal(j+1,i))

这个cal是什么呢

我们枚举能扔多少个。

然后验证,

并且验证的时候是按照什么样子匹配呢?

BZOJ 2708 [Violet 1]木偶 DP_第1张图片

有序之后就这么匹配就行辣

正确性?不会证明啊,应该看起来是对的

复杂度?n这么小谈什么复杂度。

15.10.27 Update:
对于样例 1 2 3 3 4 5的解释。
如果按照上述分段的思想。
可能会这样匹配
1 2 3 3 4 5
2 3 1 4 5 3
这样的话,木偶会剩下 3 5,而提线剩下1 3,还可以继续匹配。
并不符合上述解法。
其实只在于一个点上-》就是不同的段上的匹配方式是互逆的。
1 2 3
2 3 1 按照这种方式。
3 4 5
5 3 4 只要是上一段的逆序匹配即可,就不会出现矛盾的情况。

代码:

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define N 55
using namespace std;
int f[N];
int a[N];
int n;
int cal(int x,int y)
{
    for(int k=1;k<=y-x+1;k++)
    {
        for(int j=x;j<=y-k;j++)
            if(abs(a[j]-a[j+k])>1)return k-1;
        if(abs(a[x+k-1]-a[y-k+1])<=1)return k-1;
    }
    return y-x+1;
}
int main()
{
    while(~scanf("%d",&n))
    {
        memset(f,0,sizeof(f));
        for(int i=1;i<=n;i++)scanf("%d",&a[i]);
        sort(a+1,a+n+1);
        for(int i=1;i<=n;i++)
            for(int j=0;j<i;j++)
                f[i]=max(f[i],f[j]+cal(j+1,i));
        printf("%d\n",f[n]);
    }
}

你可能感兴趣的:(函数,dp)