一道题的不断进取!!!

区间

题目描述:

1507429531985280709.png

样例输入:

样例1:
3
1 3
2 2
1 2

样例2:
5
3 5
2 5
1 5
1 5
4 5

样例3:
20
13 16
3 17
1 6
16 20
3 13
10 13
5 19
3 20
5 7
15 16
10 16
9 20
1 13
14 16
6 11
3 4
3 11
1 14
10 11
10 13

样例输出:

样例1:
2

样例2:
3

样例3:
7

提示:

【数据规模】
对于20% 的数据, N<=20。
对于40% 的数据, N<=100。
对于60% 的数据, N<=2000。
对于80% 的数据, N<=80000。
对于100% 的数据, N<=300000。

为了强行增加难度,这道题似乎是卡常数的?

时间限制:1000ms
空间限制:256MByte

 

比赛时自己打的n*n*logn

#include<bits/stdc++.h>
#define maxn 300300
using namespace std;

int n,l=1,r,ans,ma,ce,le[maxn],re[maxn];

inline int check(int len)
{
    int tot = 0;
    for(int i=1;i<=n-len + 1;i++)
    {
        int x = i;
        int y = i + len - 1;
        int k = 0;
        for(int j=1;j<=n;j++)
        {
            if(le[j] <= x && y <= re[j]) k++;
        }
        tot = max(tot , k);
    }    
    return tot;
}

main(){
//    freopen("data.in","r",stdin);
//    freopen("C.out","w",stdout);
        
    scanf("%d",&n);
    
    for(int i=1;i<=n;i++)
    {
        scanf("%d%d",&le[i],&re[i]);
    }
    
    r = n;
    
    while(l < r)
    {
        int mid = l + r >> 1;
        ce = check(mid);
        if(ce < mid) r = mid;
        else l = mid + 1;
        ans = max(ans , min(ce , mid));
    }
    
    printf("%d",ans);    
}

 

然后学姐教了我一种n*logn*logn的做法

#include<cstdio>
#include<algorithm>
#include<cstring>
#define maxn 300300
#define lb(x) x&-x
using namespace std;
int n,l,r,ma,k,f[maxn];

struct st{
    int l,r;
}s[maxn];

bool com(st a,st b)
{
    if(a.l != b.l) return a.l < b.l;
    return a.r < b.r;
}

void up(int pos)
{
    while(pos < maxn)
    {
        f[pos]++;
        pos+=lb(pos);
    }
}

int cha(int pos)
{
    int ans = 0;
    while(pos)
    {
        ans += f[pos];
        pos -= lb(pos);
    }
    return ans;
}

int pan(int x)
{
    int d;
    int ans = 0;
    for(int i=1;i<=n;i++)
    {
        if(s[i].r - s[i].l + 1 < x) continue;
        d = s[i].l;
        up(s[i].r);
        ans = cha(maxn - 1) - cha(d + x - 2);
        if(ans >= x) return 1;
    }
    return 0;
}

main(){
    
    scanf("%d",&n);
    
    for(int i=1;i<=n;i++)
    {
        scanf("%d%d",&s[i].l,&s[i].r);
    }
    
    sort(s+1,s+1+n,com);
    
    l = 1;
    
    r = n;
    
    while(l < r)
    {
        int mid = l + r >> 1;
        memset(f,0,sizeof(f));
        if(pan(mid)) l = mid + 1;
        else r = mid;
    }
    
    printf("%d",l - 1);
}

 

最后%%%kcz%%%教了我一种n*logn的做法

#include<cstdio>
#include<cstring>
#define maxn 300300
using namespace std;
int n,l,r,ma,k,f[maxn];

struct st{
    int l,r;
}s[maxn];

int pan(int x)
{
    memset(f,0,sizeof(f));
    int ans = 0,ce = 0;
    for(int i=1;i<=n;i++)
    {
        if(s[i].r - s[i].l + 1 < x) continue;
        f[s[i].l]++;
        f[s[i].r - x + 2]--;
    }
    for(int i=1;i<=n;i++)
    {
        ans+=f[i];
        if(ans >= x) return 1;
    }
    return 0;
}

main(){
    scanf("%d",&n);
    for(int i=1;i<=n;i++) scanf("%d%d",&s[i].l,&s[i].r);
    l = 1;
    r = n;
    while(l < r)
    {
        int mid = l + r >> 1;
        if(pan(mid)) l = mid + 1;
        else r = mid;
    }
    printf("%d",l - 1);
}

 

最后时间一路飙升

 

 

最后跑得比孔爷还快!!!

还是先%为敬!!!

%%%KCZNO1%%%

 

posted @ 2017-10-09 20:36  cc123321  阅读(166)  评论(0编辑  收藏  举报