bzoj 1171: 大sz的游戏, bzoj 2892: 强袭作战

标记永久化,线段树套单调队列。

标记永久化的好处在于可以减少下传标记的时间。

如果有些标记很难下传,但是打标记的顺序对结果没有影响,就可以用标记永久化

比如说 支持矩阵加、矩阵求和就可以用带标记永久化的二维线段树来做——> 用第二维线段树维护第一维线段树上的标记

 

标记永久化具体做法:维护两种值,A、B。

修改:如果区间覆盖了一个节点,修改A,返回。如果经过一个节点,修改B,递归修改两个子树

查询:如果区间覆盖了一个节点,返回B。如果经过了一个节点,递归查询子树并加上当前节点的A值

 

另:(大雾

一维:返回一个变量的值

二维:区间求和

三维:矩形求和

四维:矩形加、矩形求和

五维:矩形加、矩形求和,带撤销

六维:矩形加、矩形求和,可持久化

七维:?

 

#include <bits/stdc++.h>
#define N 1001000
#define M 6000000
#define mp make_pair
#define fr first
#define sc second
#define pI pair <int, int>
#define INF 1000000000
#define int unsigned
using namespace std;
int n, L;
int lb[M], rb[M], lc[M], rc[M];
list <pI> tg[M], su[M];
int xi[N], yi[N], li[N], px[N];
int tot, ss, tim;
int build(int l, int r)
{
    int nw = ++ ss;
    lb[nw] = l; rb[nw] = r;
    if (l != r)
        lc[nw] = build(l, (l + r) / 2),
        rc[nw] = build((l + r) / 2 + 1, r);
    return nw;
}
void add(int t, int l, int r, pI s)
{
    if (l <= lb[t] && rb[t] <= r)
    {
        while (!tg[t].empty() && tg[t].back().sc >= s.sc) 
            tg[t].pop_back();
        tg[t].push_back(s);
        return;
    }
    
    while (!su[t].empty() && su[t].back().sc >= s.sc) 
        su[t].pop_back();
    su[t].push_back(s);
    
    if (l <= rb[lc[t]]) add(lc[t], l, r, s);
    if (r >= lb[rc[t]]) add(rc[t], l, r, s);
}
int sum(int t, int l, int r)
{
    int s;
    while (!tg[t].empty() && tg[t].front().fr < tim)
        tg[t].pop_front();
    if (!tg[t].empty()) s = tg[t].front().sc;
    else s = INF;
    
    
    while (!su[t].empty() && su[t].front().fr < tim)
        su[t].pop_front();
    if (l <= lb[t] && rb[t] <= r)
    {
        if (!su[t].empty()) s = min(s, su[t].front().sc);
        return s;
    }
    
    
    if (l <= rb[lc[t]]) s = min(s, sum(lc[t], l, r));
    if (r >= lb[rc[t]]) s = min(s, sum(rc[t], l, r));    
    return s;
}
int f[N];
signed main()
{
    scanf("%d%d", &n, &L);
    
    for (int i = 2; i <= n; ++ i)
        scanf("%d%d%d", &xi[i], &yi[i], &li[i]),
        px[++ tot] = xi[i],
        px[++ tot] = yi[i];
    
    sort(px + 1, px + tot + 1);
    xi[1] = px[1]; yi[1] = px[tot];
    tot = unique(px + 1, px + tot + 1) - px - 1;
    for (int i = 1; i <= n; ++ i)
        xi[i] = lower_bound(px + 1, px + tot + 1, xi[i]) - px,
        yi[i] = lower_bound(px + 1, px + tot + 1, yi[i]) - px;
    build(1, tot);
    
    //printf("%d\n", ss);
    add(1, 1, tot, mp(L, 0));
    for (int i = 2; i <= n; ++ i)
    {
        //if (i == 182796)
        //    puts("233");
        tim = li[i];
        f[i] = sum(1, xi[i], yi[i]) + 1;
        add(1, xi[i], yi[i], mp(li[i] + L, f[i])); 
        printf("%d\n", f[i] > n? -1: f[i]);
    }
}

 

posted @ 2017-01-18 21:19  AwD!  阅读(393)  评论(3编辑  收藏  举报