zoukankan      html  css  js  c++  java
  • 洛谷2591BZOJ2298 problem a题解

    题目连接

    bz链接

    我们发现,如果一个人有ai个分数比他高的人,有bi个分数比他低的人

    那么按照分数排序后,区间[ai+1,n-bi]中的人分数便是相同的

    这样就将一个人转化为一个区间

    也许有很多人的区间都是[x,y]所以我们令区间[x,y]的权值为这个区间的人数,即ai+1=x,n-bi=y的人的个数

    注意到权值如果大于区间长度不符合,于是要与y-x+1取max

    我们取这个区间,表示这个区间的人说的是真话

    现在问题转化为求多个不相交的区间,使得权值和最大,这样dp一下就好了

    用f[i]表示到i位置的最大权值,f[i]=max(f[l-1],f[i-1])

    # include<iostream>
    # include<cstdio>
    # include<cmath>
    # include<cstring>
    # include<algorithm>
    # include<vector>
    using namespace std;
    const int mn =  100005;
    struct edge{int to,dis,next;};
    edge e[mn];
    int head[mn],edge_max;
    void add(int x,int y,int z)
    {
        //printf("%d %d %d
    ",x,y,z);
        e[++edge_max].to=y;
        e[edge_max].dis=z;
        e[edge_max].next=head[x];
        head[x]=edge_max;
    }
    struct person{int a,b;};
    person c[mn],d[mn];
    int f[mn],tot,cnt;
    bool cmp(person x,person y)
    {
        if(x.b==y.b) return x.a<y.a;
        else return x.b<y.b;
    }
    int n;
    int main()
    {
        int x,y;
        //freopen("a.in","r",stdin);
        //freopen("a.out","w",stdout);
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
        {
            scanf("%d%d",&c[i].a,&c[i].b);
            x=c[i].a+1,y=n-c[i].b;
            if(x>y) continue;
            else d[++tot].a=x,d[tot].b=y;
        }
        sort(d+1,d+1+tot,cmp);
        /*for(int i=1;i<=tot;i++)
            printf("%d %d
    ",d[i].a,d[i].b);*/
        int now=0,N=d[tot].b;
        for(int i=1;i<=tot;i++)
        {
            now++;
            if(d[i].a!=d[i+1].a || d[i].b!=d[i+1].b){
                if(now>=d[i].b-d[i].a+1)
                    now=d[i].b-d[i].a+1;
                add(d[i].b,d[i].a,now);
                now=0;
            }
        }
        for(int i=1;i<=N;i++)
        {
            f[i]=f[i-1];
            for(int j=head[i];j;j=e[j].next)
                f[i]=max(f[i],f[e[j].to-1]+e[j].dis);
        }
        printf("%d",n-f[N]);
        return 0;
    }
  • 相关阅读:
    文件进阶
    文件及文件操作
    字符编码
    集合
    数据类型之字典
    数据类型之列表,元组
    数据类型之数字,字符串
    for 循环语句
    while 循环语句
    深浅拷贝
  • 原文地址:https://www.cnblogs.com/logeadd/p/9531006.html
Copyright © 2011-2022 走看看