BZOJ 2243 染色(树链剖分好题)

2243: [SDOI2011]染色

Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 7971  Solved: 2990
[Submit][Status][Discuss]

Description

给定一棵有n个节点的无根树和m个操作,操作有2类:

1、将节点a到节点b路径上所有点都染成颜色c;

2、询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如“112221”由3段组成:“11”、“222”和“1”。

请你写一个程序依次完成这m个操作。

Input

第一行包含2个整数n和m,分别表示节点数和操作数;

第二行包含n个正整数表示n个节点的初始颜色

下面 行每行包含两个整数x和y,表示xy之间有一条无向边。

下面 行每行描述一个操作:

“C a b c”表示这是一个染色操作,把节点a到节点b路径上所有点(包括a和b)都染成颜色c;

“Q a b”表示这是一个询问操作,询问节点a到节点b(包括a和b)路径上的颜色段数量。

Output

对于每个询问操作,输出一行答案。

Sample Input

6 5

2 2 1 2 1 1

1 2

1 3

2 4

2 5

2 6

Q 3 5

C 2 1 1

Q 3 5

C 5 1 2

Q 3 5

Sample Output

3

1

2

HINT

数N<=10^5,操作数M<=10^5,所有的颜色C为整数且在[0, 10^9]之间。

题目链接:BZOJ 2243

做了几道普通的树链剖分维护边权、点权,查询路径的题目,感觉并没有什么特点,然而这题比较有意思,求路径上连续颜色有几段,显然用线段树的话只要维护当前区间最左和最右的颜色,左右子区间即可推出父区间的答案:左边段数+右边段数-(左区间右端点颜色==右区间左端点颜色)。然后统计的时候也要利用这个思想——线段树的query与树链剖分中记录u与v上升区间段数的同时也与u、v最后上升的区间最左端点颜色比较得到答案。

代码:

#include <bits/stdc++.h>
using namespace std;
#define INF 0x3f3f3f3f
#define LC(x) (x<<1)
#define RC(x) ((x<<1)+1)
#define MID(x,y) ((x+y)>>1)
#define fin(name) freopen(name,"r",stdin)
#define fout(name) freopen(name,"w",stdout)
#define CLR(arr,val) memset(arr,val,sizeof(arr))
#define FAST_IO ios::sync_with_stdio(false);cin.tie(0);
typedef pair<int, int> pii;
typedef long long LL;
const double PI = acos(-1.0);
const int N = 100010;
struct seg
{
    int l, mid, r;
    int lc, rc;
    int s, tag;
};
struct edge
{
    int to, nxt;
    edge() {}
    edge(int _to, int _nxt): to(_to), nxt(_nxt) {}
};
edge E[N << 1];
seg T[N << 2];
int head[N], tot;
int sz[N], fa[N], son[N], top[N], dep[N], idx[N], ts;
int arr[N];
int Rc, Lc;

void init()
{
    CLR(head, -1);
    tot = 0;
    ts = 0;
}
void add(int s, int t)
{
    E[tot] = edge(t, head[s]);
    head[s] = tot++;
}
void dfs1(int u, int f, int d)
{
    sz[u] = 1;
    fa[u] = f;
    son[u] = -1;
    dep[u] = d;
    for (int i = head[u]; ~i; i = E[i].nxt)
    {
        int v = E[i].to;
        if (v != f)
        {
            dfs1(v, u, d + 1);
            sz[u] += sz[v];
            if (son[u] == -1 || sz[son[u]] < sz[v])
                son[u] = v;
        }
    }
}
void dfs2(int u, int tp)
{
    idx[u] = ++ts;
    top[u] = tp;
    if (~son[u])
        dfs2(son[u], tp);
    for (int i = head[u]; ~i; i = E[i].nxt)
    {
        int v = E[i].to;
        if (v != fa[u] && v != son[u])
            dfs2(v, v);
    }
}
void pushup(int k)
{
    T[k].s = T[LC(k)].s + T[RC(k)].s - (T[LC(k)].rc == T[RC(k)].lc);
    T[k].lc = T[LC(k)].lc;
    T[k].rc = T[RC(k)].rc;
}
void pushdown(int k)
{
    if (T[k].tag == -1)
        return ;
    T[LC(k)].tag = T[RC(k)].tag = T[k].tag;
    T[LC(k)].lc = T[LC(k)].rc = T[k].tag;
    T[RC(k)].lc = T[RC(k)].rc = T[k].tag;
    T[LC(k)].s = T[RC(k)].s = 1;
    T[k].tag = -1;
}
void build(int k, int l, int r)
{
    T[k].l = l;
    T[k].r = r;
    T[k].mid = MID(l, r);
    T[k].lc = T[k].rc = 0;
    T[k].tag = -1;
    T[k].s = 0;
    if (l == r)
        return ;
    build(LC(k), l, T[k].mid);
    build(RC(k), T[k].mid + 1, r);
}
void update(int k, int l, int r, int c)
{
    if (l <= T[k].l && T[k].r <= r)
    {
        T[k].tag = c;
        T[k].lc = T[k].rc = c;
        T[k].s = 1;
    }
    else
    {
        pushdown(k);
        if (r <= T[k].mid)
            update(LC(k), l, r, c);
        else if (l > T[k].mid)
            update(RC(k), l, r, c);
        else
        {
            update(LC(k), l, T[k].mid, c);
            update(RC(k), T[k].mid + 1, r, c);
        }
        pushup(k);
    }
}
int query(int k, int l, int r, int L, int R)
{
    if (L == T[k].l)
        Lc = T[k].lc;
    if (R == T[k].r)
        Rc = T[k].rc;
    if (l <= T[k].l && T[k].r <= r)
        return T[k].s;
    else
    {
        pushdown(k);
        if (r <= T[k].mid)
            return query(LC(k), l, r, L, R);
        else if (l > T[k].mid)
            return query(RC(k), l, r, L, R);
        else
            return query(LC(k), l, T[k].mid, L, R) + query(RC(k), T[k].mid + 1, r, L, R) - (T[LC(k)].rc == T[RC(k)].lc);
    }
}
int Find(int u, int v)
{
    int ret = 0;
    int tu = top[u], tv = top[v];
    int last_u = -1, last_v = -1;
    while (tu != tv)
    {
        if (dep[tu] < dep[tv])
        {
            swap(tu, tv);
            swap(u, v);
            swap(last_u, last_v);
        }
        ret += query(1, idx[tu], idx[u], idx[tu], idx[u]);
        if (Rc == last_u)
            --ret;
        last_u = Lc;
        u = fa[tu];
        tu = top[u];
    }
    if (dep[u] > dep[v])
    {
        swap(u, v);
        swap(last_u, last_v);
    }
    ret += query(1, idx[u], idx[v], idx[u], idx[v]);
    if (Lc == last_u)
        --ret;
    if (Rc == last_v)
        --ret;
    return ret;
}
void solve(int u, int v, int c)
{
    int tu = top[u], tv = top[v];
    while (tu != tv)
    {
        if (dep[tu] < dep[tv])
        {
            swap(tu, tv);
            swap(u, v);
        }
        update(1, idx[tu], idx[u], c);
        u = fa[tu];
        tu = top[u];
    }
    if (dep[u] > dep[v])
        swap(u, v);
    update(1, idx[u], idx[v], c);
}
int main(void)
{
    int n, m, a, b, c, i;
    char ops[10];
    while (~scanf("%d%d", &n, &m))
    {
        init();
        for (i = 1; i <= n; ++i)
            scanf("%d", &arr[i]);
        for (i = 1; i < n; ++i)
        {
            scanf("%d%d", &a, &b);
            add(a, b);
            add(b, a);
        }
        dfs1(1, 0, 1);
        dfs2(1, 1);
        build(1, 1, n);
        for (i = 1; i <= n; ++i)
            update(1, idx[i], idx[i], arr[i]);
        while (m--)
        {
            scanf("%s", ops);
            if (ops[0] == ‘Q‘)
            {
                scanf("%d%d", &a, &b);
                printf("%d\n", Find(a, b));
            }
            else
            {
                scanf("%d%d%d", &a, &b, &c);
                solve(a, b, c);
            }
        }
    }
    return 0;
}
时间: 2024-08-06 07:58:23

BZOJ 2243 染色(树链剖分好题)的相关文章

BZOJ 2243 染色 | 树链剖分模板题进阶版

BZOJ 2243 染色 | 树链剖分模板题进阶版 这道题呢~就是个带区间修改的树链剖分~ 如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了. 这道题要注意的是,无论是线段树上还是原树上,把两个区间的信息合并的时候,要注意中间相邻两个颜色是否相同. 这代码好长啊啊啊啊 幸好一次过了不然我估计永远也De不出来 #include <cstdio> #include <cstring> #include <algorithm> using namesp

bzoj 2243 染色(树链剖分)

题外话 首先这是个挺裸的题,由于太久没写剖分导致调了好久,前天调了一下午,一直查不到错 昨晚在看春晚的时候突然灵机一动,发现合并的时候出了问题,开电脑把它A掉了= = 感觉自己也蛮拼的 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1:将节点a到节点b路径上所有点都染成颜色c 2:询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如"112221"由3段组成:"11"."222"和"1&q

hysbz 2243 染色(树链剖分)

题目链接:hysbz 2243 染色 题目大意:略. 解题思路:树链剖分+线段树的区间合并,但是区间合并比较简单,节点只要记录左右端点的颜色即可. #include <cstdio> #include <cstring> #include <algorithm> using namespace std; const int maxn = 1e5 + 5; int N, M, ne, val[maxn], first[maxn], jump[maxn * 2]; int

HYSBZ 2243 染色 树链剖分 线段树

这题对最简单的树链剖分做了一些变化,在链的转移过程中要考虑前后链相邻节点颜色是否相同. 对于线段树,只要维护三个值,左端点颜色,右端点颜色还有区间颜色总数就好了. #include <cstdio> #include <cstring> #include <algorithm> #include <map> #include <set> #include <bitset> #include <queue> #include

BZOJ 2243: [SDOI2011]染色 树链剖分

2243: [SDOI2011]染色 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1886  Solved: 752[Submit][Status] Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如“112221”由3段组成:“11”.“222”和“1”. 请你写一个程序依次完成这m个操作. In

HDU 2966 Aragorn&#39;s Story 树链剖分第一题 基础题

Problem Description Our protagonist is the handsome human prince Aragorn comes from The Lord of the Rings. One day Aragorn finds a lot of enemies who want to invade his kingdom. As Aragorn knows, the enemy has N camps out of his kingdom and M edges c

HDU 3966 Aragorn&#39;s Story(树链剖分 模板题)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3966 Problem Description Our protagonist is the handsome human prince Aragorn comes from The Lord of the Rings. One day Aragorn finds a lot of enemies who want to invade his kingdom. As Aragorn knows, th

BZOJ 3083 遥远的国度 树链剖分

3083: 遥远的国度 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 797  Solved: 181[Submit][Status] Description 描述 zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcwwzdjn完成任务后才能进入遥远的国度继续追杀. 问题是这样的:遥远的国度有n个城市,这些城市之间由一些路连

Bzoj1036 树链剖分基础题

树链剖分的基础题 因为复习到了这个部分突然发现竟然没有题解所以现在补一个.. 一些基础的东西... 重儿子:siz[u]为v的子节点中siz值最大的,那么u就是v的重儿子. 轻儿子:v的其它子节点. 重边:点v与其重儿子的连边. 轻边:点v与其轻儿子的连边. 重链:由重边连成的路径. 轻链:轻边. 然后简单的用两次dfs计算出每个节点的father,deep,size, son,w,top 其他简单的就不说了 w表示的是当前节点与其付清节点的连边在线段树中的位置 top表示的是当前节点所在的链的