星星之火

[JXOI 2018] 守卫 解题报告 (DP)

interlinkage:

https://www.luogu.org/problemnew/show/P4563

description:

solution:

  • 注意到对于范围$[l,r]$,$r$这个亭子一定要设置保镖
  • 设$f_{l,r}$为活动范围为$[l,r]$所需的最小保镖数
  • 枚举右端点$r$,然后从右向左遍历左端点$l$。设$p$为在$[l,r]$中$r$处能看到的最左的点
  • 这样的话我们就必须在$p$或者$p-1$处放置保镖
  • 于是$f_{l,r}=min(f_{l,p-1},f_{l,p})+f_{p+1,r}$
  • 注意一下$p$的更新,若$l$到$r$连线的斜率小于$p$到$r$连线的斜率,那么就要更新$p=l$了

code:

#include<algorithm>
#include<cstring>
#include<cstdio>
#include<iostream>
#include<cmath>
using namespace std;

const int N=5000+15;
int n;
int h[N],f[N][N];
inline int read()
{
    char ch=getchar();int s=0,f=1;
    while (ch<'0'||ch>'9') {if (ch=='-') f=-1;ch=getchar();}
    while (ch>='0'&&ch<='9') {s=(s<<3)+(s<<1)+ch-'0';ch=getchar();}
    return s*f;
}
double slope(int l,int r) {return (double)(h[r]-h[l])/(r-l);}
bool cansee(int l,int p,int r) 
{
    return slope(l,r)<slope(p,r);
}
int main()
{
    n=read();
    for (int i=1;i<=n;i++) h[i]=read();
    int ans=0;
    for (int r=1;r<=n;r++)
    {
        f[r][r]=1;ans^=f[r][r];
        int p=0;
        for (int l=r-1;l>=1;l--)
        {
            if (!p||cansee(l,p,r)) p=l;
            f[l][r]=min(f[l][p-1],f[l][p])+f[p+1][r];
            ans^=f[l][r];
        }
    }
    printf("%d\n",ans);
    return 0;
}
posted @ 2019-04-14 16:38  星星之火OIer  阅读(193)  评论(0编辑  收藏  举报