ACM-T3分块

                                                 ty的难题

题目背景

      国民男神ty又遇到了一个小难题,他在和xqj大神的争论中(谁更强),ty表示自己不会这个问题(装弱),于是他将这个问题交给了身为ty小迷弟(妹)的你。

题目描述:给一个长为n的数列,以及n次操作。每次操作均有一串字符和3个数字组成(c,l,r,x);有两种操作:将区间l~r加上x,询问区间l,r中比x小的

最大元素(若不存在则输出impossible),对应每个操作,字符分别为‘change’,‘query’。

样例输入

3

1 2 3

query 1 2 2

change 1 2 1

query 1 2 2

样例输出

1

impossible

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<set>
using namespace std;
int blo,bl[100005],v[100005],n,lazy[1005];
long long read()
{
    long long x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    return x*f;
}
set<int> st[1005];
void clear(int x)
{
    st[x].clear();
    for(int i=(x-1)*blo+1;i<=min(n,x*blo);i++)
    st[x].insert(v[i]);
    //sort(st[x].begin(),st[x].end());
}
void change(int l,int r,int val)
{
    for(int i=l;i<=min(bl[l]*blo,r);i++)
    v[i]+=val;
    clear(bl[l]);
    if(bl[l]!=bl[r])
    {
        for(int i=r;i>=(bl[r]-1)*blo+1;i--)
        v[i]+=val;
        clear(bl[r]);
    }
    for(int i=bl[l]+1;i<=bl[r]-1;i++)
    lazy[i]+=val;
}
void query(int l,int r,int x)
{
    int ans=-1;
    for(int i=l;i<=min(bl[l]*blo,r);i++)
    if(v[i]+lazy[bl[l]]<x)
    ans=max(ans,v[i]+lazy[bl[l]]);
    if(bl[l]!=bl[r])
    {
        for(int i=r;i>=(bl[r]-1)*blo+1;i--)
        if(v[i]+lazy[bl[r]]<x)
        ans-max(ans,v[i])+lazy[bl[r]];
    }
    for(int i=bl[l]+1;i<=bl[r]-1;i++)
    {
        int c=x-lazy[i];
        set<int>::iterator it1=st[i].lower_bound(c);
        if(st[i].begin()==it1) continue;
        it1--;
        ans=max(ans,*it1+lazy[i]);
    }
    if(ans==-1) printf("impossible\n");else
    printf("%d\n",ans);
}
int main()
{    n=read();
    blo=sqrt(n);
    for(int i=1;i<=n;i++)
    v[i]=read(),bl[i]=(i-1)/blo+1,st[bl[i]].insert(v[i]);
    //for(int i=1;i<=bl[n];i++)
    //sort(st[i].begin(),st[i].end());
    char char_[10];
    for(int i=1;i<=n;i++)
    {
        scanf("%s",&char_);
        if(char_[0]=='c')
        {
            int l,r,val;
            scanf("%d %d %d",&l,&r,&val);
            change(l,r,val);
        }else{
            int l,r,x;
            scanf("%d %d %d",&l,&r,&x);
            query(l,r,x);
        }
    }
    return 0;
}

 这个常数有点大,1100000的数据得3秒。

每一次clear都更新整个块

用set的性质可以直接改(删掉原来的,加上新的)

但是这种写法貌似在这种题目里有问题(可能在一个块里前面有一个2,后面有一个2,现在将后面那个2改为3,在set中前面那个2会变成3),所以题解有问题!

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<set>
using namespace std;
int blo,bl[100005],v[100005],n,lazy[1005];
long long read()
{
    long long x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    return x*f;
}
set<int> st[1005];    
void change(int l,int r,int val)
{
for(int i=l;i<=min(bl[l]*blo,r);i++)
{
st[bl[l]].erase(v[i]);
v[i]+=val;
st[bl[r]].insert(v[i]);
} 
if(bl[l]!=bl[r])
{
    for(int i=r;i>=(bl[r]-1)*blo+1;i--)
    {
        st[bl[r]].erase(v[i]);
        v[i]+=val;
        st[bl[r]].insert(v[i]);
    }
}    
for(int i=bl[l]+1;i<=bl[r]-1;i++)
lazy[i]+=val;
}
void query(int l,int r,int x)
{
    int ans=-1;
    for(int i=l;i<=min(bl[l]*blo,r);i++)
    if(v[i]+lazy[bl[l]]<x)
        ans=max(ans,v[i]+lazy[bl[l]]);
    if(bl[l]!=bl[r])
    {    
        for(int i=r;i>=(bl[r]-1)*blo+1;i--)
        if(v[i]+lazy[bl[r]]<x)
        ans=max(ans,v[i])+lazy[bl[r]];
    }
for(int i=bl[l]+1;i<=bl[r]-1;i++)
{
int c=x-lazy[i];
set<int>::iterator it1=st[i].lower_bound(c);
if(st[i].begin()==it1) continue;
it1--;
ans=max(ans,*it1+lazy[i]);
}
if(ans==-1) printf("impossible\n");else
printf("%d\n",ans);
}
int main()
{ n=read();
blo=sqrt(n);
for(int i=1;i<=n;i++)
v[i]=read(),bl[i]=(i-1)/blo+1,st[bl[i]].insert(v[i]);
//for(int i=1;i<=bl[n];i++)
//sort(st[i].begin(),st[i].end());
char char_[10];
for(int i=1;i<=n;i++)
{
scanf("%s",&char_);
if(char_[0]=='c')
{
int l,r,val;
scanf("%d %d %d",&l,&r,&val);
change(l,r,val);
}else{
int l,r,x;
scanf("%d %d %d",&l,&r,&x);
query(l,r,x);
}
}
return 0;
} 

 但是这样的话set的性质大部分已经不用,直接用vector常数更小

#include<map>
#include<set>
#include<cmath>
#include<stack>
#include<queue>
#include<cstdio>
#include<vector>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define mod 998244353
#define pi acos(-1)
#define inf 0x7fffffff
#define ll long long
using namespace std;
ll read()
{
    ll x=0,f=1;char ch=getchar();
    while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
    return x*f;
}
int n,blo;
int v[100005],bl[100005],atag[100005];
set<int>st[105];
void add(int a,int b,int c)
{
    for(int i=a;i<=min(bl[a]*blo,b);i++)
    {
        v[i]+=c;
    }
    st[bl[a]].clear();
    for(int i=(bl[a]-1)*blo+1;i<=min(bl[a]*blo,n);i++)
    st[bl[a]].insert(v[i]);
    if(bl[a]!=bl[b])
    {
        for(int i=(bl[b]-1)*blo+1;i<=b;i++)
        {
            v[i]+=c;  
        }
        st[bl[b]].clear();
        for(int i=(bl[b]-1)*blo+1;i<=min(bl[b]*blo,n);i++)
        st[bl[b]].insert(v[i]);
    }
    for(int i=bl[a]+1;i<=bl[b]-1;i++)
        atag[i]+=c;
}
int query(int a,int b,int c)
{
    int ans=-1;
    for(int i=a;i<=min(bl[a]*blo,b);i++)
    {
        int val=v[i]+atag[bl[a]];
        if(val<c)ans=max(val,ans);
    }
    if(bl[a]!=bl[b])        
        for(int i=(bl[b]-1)*blo+1;i<=b;i++)        
        {
            int val=v[i]+atag[bl[b]];
            if(val<c)ans=max(val,ans);
        }
    for(int i=bl[a]+1;i<=bl[b]-1;i++)
    {
        int x=c-atag[i];
        set<int>::iterator it=st[i].lower_bound(x);
        if(it==st[i].begin())continue;
        --it;
        ans=max(ans,*it+atag[i]);
    }
    return ans;
}
int main()
{
    n=read();blo=1000;
    for(int i=1;i<=n;i++)v[i]=read();
    for(int i=1;i<=n;i++)
    {
        bl[i]=(i-1)/blo+1;
        st[bl[i]].insert(v[i]);
    }
    for(int i=1;i<=n;i++)
    {
        int f=read(),a=read(),b=read(),c=read();
        if(f==0)add(a,b,c);
        if(f==1)printf("%d\n",query(a,b,c));
    }
    return 0;
}

 

posted @ 2017-05-20 13:44  dancer16  阅读(244)  评论(1编辑  收藏  举报
描述