CF1045G AI robots

题目大意就不说了

这道题可以用CDQ分治做

但是,如何选择CDQ分治的维度一直是CDQ分治的难点所在

这道题我们有三种选择

1.让智商高的数智商低的

2.让看的近的数看的远的

3.让靠右的数靠左的

但是,1和3都不好满足让这两个机器人分别都能看到的要求(因为不保证所以的机器人的视野范围相同)

所以我们以其视野范围当做第一维(从大到小排序)

之后我们发现对于左边\(i\)

右边满足其智商限制的一定是一个区间,之后这个区间我们可以用双指针维护

之后将这个区间里的数加入权值树状数组查询

另外由于至于范围很大

我们要离散化,但是离散化\(a_i-r_i\)\(a_i+r_i\)也应当离散化

所以数组开三倍,之前因为这个RE了

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cctype>
using namespace std;
const int N = 5e5 + 3;
struct node{
    int xi;
    int ri;
    int qi;
}a[N];
int b[N];
int n,k;
long long ans;
struct BIT{
    int c[N << 1];
    inline void ins(int x,int v){for(;x <= b[0];x += x & -x) c[x] += v;}
    inline int query(int x){
        int res = 0;
        for(;x;x -= x & -x) res += c[x];
        return res; 
    }
}T;
inline bool cmp1(node x,node y){
    if(x.ri != y.ri) return x.ri > y.ri;
    if(x.qi != y.qi) return x.qi < y.qi;
    return x.xi < y.xi;
}
inline bool cmp2(node x,node y){
    if(x.qi != y.qi) return x.qi < y.qi;
    return x.xi < y.xi; 
}
inline int read(){
    int v = 0,c = 1;char ch = getchar();
    while(!isdigit(ch)){
        if(ch == '-') c = -1;
        ch = getchar();
    }
    while(isdigit(ch)){
        v = v * 10 + ch - 48;
        ch = getchar(); 
    }
    return v * c;
}
inline void solve(int l,int r){
    if(l == r) return ;
    int mid = (l + r) >> 1;
    solve(l,mid);solve(mid + 1,r);
    sort(a + l,a + mid + 1,cmp2);sort(a + mid + 1,a + r + 1,cmp2);
    int L = l,R = l - 1;
    for(int i = mid + 1;i <= r;++i){
        while(R + 1 <= mid && a[R + 1].qi <= a[i].qi + k) T.ins(a[R + 1].xi,1),R++; 
        while(L <= mid && a[L].qi < a[i].qi - k) T.ins(a[L].xi,-1),L++;
        int from = lower_bound(b + 1,b + b[0] + 1,b[a[i].xi] - a[i].ri) - b;
        int to = lower_bound(b + 1,b + b[0] + 1,b[a[i].xi] + a[i].ri) - b;
        ans += T.query(to) - T.query(from - 1);
    }
    for(int i = L;i <= R;++i) T.ins(a[i].xi,-1);
}
int main(){
    n = read(),k = read();
    for(int i = 1;i <= n;++i){
        a[i].xi = read();
        a[i].ri = read();
        a[i].qi = read();
        b[++b[0]] = a[i].xi - a[i].ri;
        b[++b[0]] = a[i].xi + a[i].ri;
        b[++b[0]] = a[i].xi;
    }
    sort(b + 1,b + b[0] + 1);
    b[0] = unique(b + 1,b + b[0] + 1) - b - 1;
    for(int i = 1;i <= n;++i) a[i].xi = lower_bound(b + 1,b + b[0] + 1,a[i].xi) - b;
    sort(a + 1,a + n + 1,cmp1);
    solve(1,n);
    printf("%I64d\n",ans);
    return 0;   
}