BZOJ3236: [Ahoi2013]作业 莫队+分块

3236: [Ahoi2013]作业

Time Limit: 100 Sec   Memory Limit: 512 MB
Submit: 1272   Solved: 490

Description

Input

Output

Sample Input

3 4
1 2 2
1 2 1 3
1 2 1 1
1 3 1 3
2 3 2 3

Sample Output

2 2
1 1
3 2
2 1

HINT


N=100000,M=1000000

题解:
莫队+分块 时间复杂度O(n√n)

#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define N 100010
#define M 1000010
using namespace std;
int a[N],p;
struct pp {int l,r,a,b,num;}b[M];
int bl[N],L[1001],R[1001];
int tot,ans[N],ansv[M],ansn[M];
int tn[1001],tv[1001];
bool cmp(pp x,pp y)
{
    if(bl[x.l]!=bl[y.l]) return bl[x.l]<bl[y.l];
    return x.r<y.r;
}
void add(int x)
{
    ans[x]++;
	tv[bl[x]]++;
    if(ans[x]==1) tn[bl[x]]++;
}
void del(int x)
{
    ans[x]--;
	tv[bl[x]]--;
    if(ans[x]==0) tn[bl[x]]--;
}
int Query_value(int a,int b)
{
    int ka=bl[a],kb=bl[b];
    int ret=0;
    for(int i=ka+1;i<=kb-1;i++) ret+=tv[i];
    if(ka==kb)
    {
        for(int i=a;i<=b;i++) ret+=ans[i];
    }
    else
    {
        for(int i=a;i<=R[ka];i++) ret+=ans[i];
        for(int i=L[kb];i<=b;i++) ret+=ans[i];
    }
    return ret;
}
int Query_num(int a,int b)
{
    int ka=bl[a],kb=bl[b];
    int ret=0;
    for(int i=ka+1;i<=kb-1;i++) ret+=tn[i];
    if(ka==kb)
    {
        for(int i=a;i<=b;i++)
        if(ans[i]) ret++;
    }
    else
    {
        for(int i=a;i<=R[ka];i++)
        if(ans[i]) ret++;
        for(int i=L[kb];i<=b;i++)
        if(ans[i]) ret++;
    }
    return ret;
}
int n,m,k;
int main()
{
    scanf("%d%d",&n,&m);
    int block=sqrt(n);
    for(int i=1;i<=n;i++) scanf("%d",&a[i]);
    for(int i=1;i<=n;i++) bl[i]=(i-1)/block+1;
    for(int i=1;i<=n;i++)
    {
        R[bl[i]]=i;
        if(!L[bl[i]]) L[bl[i]]=i;
    }
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d%d%d",&b[i].l,&b[i].r,&b[i].a,&b[i].b);
        b[i].num=i;
    }
    sort(b+1,b+m+1,cmp);
    int l=1,r=0;
    for(int i=1;i<=m;i++)
    {
        for(;r<b[i].r;r++) add(a[r+1]);
        for(;r>b[i].r;r--) del(a[r]);
        for(;l<b[i].l;l++) del(a[l]);
        for(;l>b[i].l;l--) add(a[l-1]);
        ansv[b[i].num]=Query_value(b[i].a,b[i].b);
        ansn[b[i].num]=Query_num(b[i].a,b[i].b);
    }
    for(int i=1;i<=m;i++) printf("%d %d\n",ansv[i],ansn[i]);
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值