• LOJ2255. 「SNOI2017」炸弹 (线段树)


    本文为线段树做法

    (听说可以tarjan缩点+拓扑?
    感觉差不多。。而且这样看起来方便很多

    找到左端点的过程可以看作
    点 -> 区间内lowerbound最小的点 -> lowerbound -> 区间内lowerbound最小的点 -> lowerbound -> ......

    所以直接维护每个点lowerbound,线段树维护下就好啦

    右端点同理

    #include <cstdio>
    #include <algorithm>
    #include <cmath>
    #include <cstring>
    using namespace std;
    const int N = 5e5 + 5;
    const long long P = 1e9 + 7;
    const int inf = 0x3f3f3f3f;
    
    int n, m;
    long long pos[N], rad[N];
    int L[N], R[N];
    long long ans;
    
    struct Seg{
    	int lm[N << 2], rm[N << 2];
    	void update(int rt){
    		lm[rt] = min(lm[rt << 1], lm[rt << 1 | 1]);
    		rm[rt] = max(rm[rt << 1], rm[rt << 1 | 1]);
    	}
    	void build(int l, int r, int rt){
    		if(l == r){
    			lm[rt] = L[l], rm[rt] = R[l]; 
    			return ;
    		}
    		int mid = l + ((r - l) >> 1);
    		build(l, mid, rt << 1);
    		build(mid + 1, r, rt << 1 | 1);
    		update(rt);
    	}
    	void qry(int l, int r, int x, int y, int& nx, int& ny, int rt){
    		if(l == x && r == y){
    			nx = min(lm[rt], nx); ny = max(rm[rt], ny); return ;
    		}
    		int mid = l + ((r - l) >> 1);
    		if(x <= mid) qry(l, mid, x, min(y, mid), nx, ny, rt << 1);
    	        if(y > mid) qry(mid + 1, r, max(x, mid + 1), y, nx, ny, rt << 1 | 1);
    	}
    }seg;
    
    inline int l_lim(int x){
    	return lower_bound(pos + 1, pos + x, pos[x] - rad[x]) - pos;
    }
    
    inline int r_lim(int x){
    	return upper_bound(pos + x + 1, pos + n + 1, pos[x] + rad[x]) - pos - 1;
    }
    
    int main(){
    	scanf("%d", &n);
    	for(int i = 1; i <= n; ++i){
    		scanf("%lld%lld", &pos[i], &rad[i]);
    	}
    	for(int i = 1; i <= n; ++i){
    		L[i] = l_lim(i);
    		R[i] = r_lim(i);
    	}
    	seg.build(1, n, 1);
    	
    	ans = 0;
    	int x, y, nx, ny;
    	for(int i = 1; i <= n; ++i){
    	    x = y = nx = ny = i;
    	    do{
    	    	x = nx, y = ny;
    	        nx = inf, ny = -inf; seg.qry(1, n, x, y, nx, ny, 1);  
    	    }while((nx ^ x) | (ny ^ y));
    	    ans = (ans + 1ll * i * (y - x + 1) % P) % P;
    	    //这里原来忘乘1ll了爆了long long
    	}
    	printf("%lld
    ", ans);
        return 0;	
    }
    /*
    单纯维护两边并更新不可行 可能会有折向引爆 
    */
    
  • 相关阅读:
    最小移动次数使数组元素相等
    计算几何
    北校门外的回忆
    洗衣服
    HDU1046:Gridland
    注册机的实现如(istarMenu CORE Keygen)
    是你吗?
    (译)KVO的内部实现
    通过UIColor转换为UIImage
    Palindrome Number 解题报告
  • 原文地址:https://www.cnblogs.com/hjmmm/p/10473164.html
Copyright © 2020-2023  润新知