[set]JZOJ 5821 手机信号

Description
 
Input
第一行由一个空格隔开的两个正整数 m, c,意义见题目描述。
接下来 m 行,每行可能有以下形式:
construct l r v 代表发生了第一种事件;
destruct l r 代表发生了第二种事件;
query x 代表发生了第三种事件。 
Output
对于每个 query 操作,请输出一行一个整数代表此时坐标 x 处的信号强度。
 
Sample Input
11 10000
query 5
construct 5 500 100
query 500
query 1000
construct 10 90 5 
query 44
destruct 44 66
query 55
construct 50 60 3
query 46
query 6000
Sample Output
0
975
0
9999
9775
9984
0 
 
Data Constraint

分析

用set维护三元组(l,r,v),然后对于新加入的区间和原期间重叠时,劈开与新区间相交或重合的部分即可

然后删除类似,查询有在区间内和区间外两种情况

这是第一次用set写题呢

#include <iostream>
#include <cstdio>
#include <set>
#include <cstring>
using namespace std;
struct Ivs {
    int l,r,v;
    bool operator < (const Ivs a) const {
        return l<a.l||l==a.l&&r<a.r||l==a.l&&r==a.r&&v<a.v;
    }
};
set<Ivs> s;
long long c;
int m;
char order[30];

void Insert(int l,int r,int v) {
    s.insert((Ivs){l,r-(r-l)%v,v});
    set<Ivs>::iterator iter=s.lower_bound((Ivs){l,r-(r-l)%v,v}),p=iter;
    p--;
    if (iter!=s.begin()) {
        Ivs pstar=*p;
        if (pstar.r>=l) {
            s.erase(p);
            if (l-1>=pstar.l) 
            s.insert((Ivs){pstar.l,l-1-(l-1-pstar.l)%pstar.v,pstar.v});
            if (r+1<=pstar.r)
            s.insert((Ivs){r+1+((r+1-pstar.l)%pstar.v?pstar.v-(r+1-pstar.l)%pstar.v:0),pstar.r,pstar.v});
        }
    }
}

void Destroy(int l,int r) {
    s.insert((Ivs){l,r,-1});
    set<Ivs>::iterator iter=s.lower_bound((Ivs){l,r,-1}),p=iter,n=iter,rq;
    p--;n++;
    if (iter!=s.begin()) {
        Ivs pstar=*p;
        if (pstar.r>=l) {
            s.erase(p);
            if (l-1>=pstar.l) 
            s.insert((Ivs){pstar.l,l-1-(l-1-pstar.l)%pstar.v,pstar.v});
            if (r+1<=pstar.r)
            s.insert((Ivs){r+1+((r+1-pstar.l)%pstar.v?pstar.v-(r+1-pstar.l)%pstar.v:0),pstar.r,pstar.v});
        }
    }
    while (n!=s.end()) {
        Ivs nstar=*n;rq=n;n++;
        if (nstar.l<=r) {
            s.erase(rq);
            if (r+1<=nstar.r)
            s.insert((Ivs){r+1+((r+1-nstar.l)%nstar.v?nstar.v-(r+1-nstar.l)%nstar.v:0),nstar.r,nstar.v});
        }
        else break;
    }
    s.erase((Ivs){l,r,-1});
}

long long Query(int x) {
    long long mx=1e9+1;
    s.insert((Ivs){x,x,-1});
    set<Ivs>::iterator iter=s.lower_bound((Ivs){x,x,-1}),p=iter,n=iter;
    p--;n++;
    if (iter!=s.begin()) {
        Ivs pstar=*p;
        mx=min(mx,
        (long long)(x<=pstar.r?min((x-pstar.l)%pstar.v,pstar.v-(x-pstar.l)%pstar.v)
        :x-pstar.r));
    }
    if (n!=s.end()) mx=min(mx,(long long)(*n).l-x);
    s.erase((Ivs){x,x,-1});
    return max(0ll,c-mx*mx);
}

int main() {
    freopen("cellphone.in","r",stdin);
    freopen("cellphone.out","w",stdout);
    scanf("%d%lld",&m,&c);
    while (m--) {
        int l,r,x;
        scanf("%s%d",&order,&l);
        if (order[0]=='q') printf("%lld\n",Query(l));
        if (order[0]=='c') {
            scanf("%d%d",&r,&x);
            Insert(l,r,x);
        }
        if (order[0]=='d') {
            scanf("%d",&r);
            Destroy(l,r);
        }
    }
    fclose(stdin);fclose(stdout);
}
View Code

 

转载于:https://www.cnblogs.com/mastervan/p/9495041.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值