#include <cstdio>
#include <cstring>
#include <cstdlib>
using namespace std;

const int N=100001;

int n,m,color[N];   //Basic
struct G
{
    int v,nxt;
}map[N+N];
int hd[N],tt;   //Graph
int son[N],dep[N],pre[N],size[N];
int tid[N],top[N],num;  //Chain
struct T
{
    int l,r;
    int lc,rc,tag;
    int cnt;
}tr[N<<2];    //Segment Tree

inline int read(void)
{
    int s=0,f=1; char c=getchar();
    for (;c<'0'||c>'9';c=getchar()) if (c=='-') f=-1;
    for (;'0'<=c&&c<='9';c=getchar()) s=(s<<1)+(s<<3)+c-'0';
    return s*f;
}

inline void ins(int u,int v)
{
    map[++tt].v=v;
    map[tt].nxt=hd[u];
    hd[u]=tt;
}

void find(int now,int ht)
{
    dep[now]=ht,size[now]=1;
    for (int k=hd[now];k;k=map[k].nxt)
        if (!size[map[k].v])
        {
            find(map[k].v,ht+1);
            size[now]+=size[map[k].v];
            pre[map[k].v]=now;
            if (!son[now]||size[son[now]]<size[map[k].v]) son[now]=map[k].v;
        }
}

void cut(int now,int anc)
{
    top[now]=anc,tid[now]=++num;
    if (son[now]) cut(son[now],anc);
    for (int k=hd[now];k;k=map[k].nxt)
        if (!tid[map[k].v]) cut(map[k].v,map[k].v);
}

inline void clear(int now)
{
    if (tr[now].tag==-1) return;
    tr[now<<1].tag=tr[now<<1|1].tag=tr[now].tag;
    tr[now<<1].lc=tr[now<<1].rc=tr[now].tag;
    tr[now<<1|1].lc=tr[now<<1|1].rc=tr[now].tag;
    tr[now<<1].cnt=tr[now<<1|1].cnt=1;
    tr[now].tag=-1;
}

void build(int now,int l,int r)
{
    tr[now].l=l;
    tr[now].r=r;
    tr[now].tag=-1;
    if (l^r)
    {
        int mid=l+r>>1;
        build(now<<1,l,mid);
        build(now<<1|1,mid+1,r);
    }
}

void ins(int now,int l,int r,int c)
{
    if (l<=tr[now].l&&tr[now].r<=r)
    {
        tr[now].tag=tr[now].lc=tr[now].rc=c;
        tr[now].cnt=1;
        return;
    }
    clear(now);

    int mid=tr[now].l+tr[now].r>>1;
    if (l<=mid) ins(now<<1,l,r,c);
    if (mid<r) ins(now<<1|1,l,r,c);

    tr[now].lc=tr[now<<1].lc;
    tr[now].rc=tr[now<<1|1].rc;
    tr[now].cnt=tr[now<<1].cnt+tr[now<<1|1].cnt-(tr[now<<1].rc==tr[now<<1|1].lc);
}

void init(void)
{
    n=read(),m=read();
    for (int i=1;i<=n;i++) color[i]=read();

    int u,v;
    for (int i=1;i<n;i++)
    {
        u=read(),v=read();
        ins(u,v),ins(v,u);
    }

    find(1,1);
    cut(1,1);

    build(1,1,n);
    for (int i=1;i<=n;i++) ins(1,tid[i],tid[i],color[i]);
}

inline int LCA(int x,int y)
{
    for (;top[x]^top[y];)
        if (dep[top[x]]>dep[top[y]])
            x=pre[top[x]];
        else y=pre[top[y]];
    return dep[x]<dep[y]?x:y;
}

int query(int now,int l,int r)
{
    if (l<=tr[now].l&&tr[now].r<=r) return tr[now].cnt;
    clear(now);
    int mid=tr[now].l+tr[now].r>>1,sum=0;
    if (l<=mid) sum+=query(now<<1,l,r);
    if (mid<r) sum+=query(now<<1|1,l,r);
    if (l<=mid&&mid<r) sum-=tr[now<<1].rc==tr[now<<1|1].lc;
    return sum;
}

int findcolor(int now,int loc)
{
    if (tr[now].l==tr[now].r) return tr[now].tag;
    clear(now);
    int mid=tr[now].l+tr[now].r>>1;
    return findcolor(now<<1|(loc<=mid?0:1),loc);
}

void work(void)
{
    char c; int x,y,z,p,res,uc,dc;
    for (int i=1;i<=m;i++)
    {
        scanf("\n%c",&c);
        if (c=='C')
        {
            x=read(),y=read(),z=read(),p=LCA(x,y);
            for (;top[x]^top[p];x=pre[top[x]]) ins(1,tid[top[x]],tid[x],z);
            for (;top[y]^top[p];y=pre[top[y]]) ins(1,tid[top[y]],tid[y],z);
            ins(1,tid[p],x^p?tid[x]:tid[y],z);
        }
        else
        {
            x=read(),y=read(),p=LCA(x,y);
            uc=dc=-1,res=0;
            for (;top[x]^top[p];x=pre[top[x]])
            {
                if (uc^-1)
                {
                    dc=findcolor(1,tid[x]);
                    res-=uc==dc;
                }
                res+=query(1,tid[top[x]],tid[x]);
                uc=findcolor(1,tid[top[x]]);
            }
            dc=findcolor(1,tid[x]);
            res-=uc==dc;
            uc=dc=-1;
            for (;top[y]^top[p];y=pre[top[y]])
            {
                if (uc^-1)
                {
                    dc=findcolor(1,tid[y]);
                    res-=uc==dc;
                }
                res+=query(1,tid[top[y]],tid[y]);
                uc=findcolor(1,tid[top[y]]);
            }
            dc=findcolor(1,tid[y]);
            res-=uc==dc;
            res+=query(1,tid[p],x^p?tid[x]:tid[y]);
            printf("%d\n",res);
        }
    }
}

int main(void)
{
    init();
    work();

    return 0;
}

【BZOJ 2243】染色 - 树链剖分+线段树的更多相关文章

  1. 2243: [SDOI2011]染色 树链剖分+线段树染色

    给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段), 如“112221”由3段组 ...

  2. bzoj2243[SDOI2011]染色 树链剖分+线段树

    2243: [SDOI2011]染色 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 9012  Solved: 3375[Submit][Status ...

  3. 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树

    [BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...

  4. B20J_2243_[SDOI2011]染色_树链剖分+线段树

    B20J_2243_[SDOI2011]染色_树链剖分+线段树 一下午净调这题了,争取晚上多做几道. 题意: 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成 ...

  5. BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )

    BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...

  6. BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)

    BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...

  7. BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)

    前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...

  8. bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)

    4196: [Noi2015]软件包管理器 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 2852  Solved: 1668[Submit][Sta ...

  9. BZOJ2243 [SDOI2011]染色(树链剖分+线段树合并)

    题目链接 BZOJ2243 树链剖分 $+$ 线段树 线段树每个节点维护$lc$, $rc$, $s$ $lc$代表该区间的最左端的颜色,$rc$代表该区间的最右端的颜色 $s$代表该区间的所有连续颜 ...

  10. bzoj 2157: 旅游【树链剖分+线段树】

    裸的树链剖分+线段树 但是要注意一个地方--我WA了好几次才发现取完相反数之后max值和min值是要交换的-- #include<iostream> #include<cstdio& ...

随机推荐

  1. SqlSever基础 group by之后,加having 对分组之后的数据在进行处理

    镇场诗:---大梦谁觉,水月中建博客.百千磨难,才知世事无常.---今持佛语,技术无量愿学.愿尽所学,铸一良心博客.------------------------------------------ ...

  2. Mybatis+struts2+spring整合

    把student项目改造成ssm  struts2 +mybatis+spring 1,先添加spring支持:类库三个,applicationContext.xml写在webinf下四个命名空间,监 ...

  3. mysql密码忘记或者不知道,怎么办?

    运行cmd: 输入mysql回车,如果成功,将出现MySQL提示符 > 连接权限数据库>use mysql; (>是本来就有的提示符,别忘了最后的分号) 修改改密码:> upd ...

  4. 2016.3.22考试(HNOI难度)

    T1 盾盾的打字机 盾盾有一个非常有意思的打字机,现在盾哥要用这台打字机来打出一段文章. 由于有了上次的经验,盾盾预先准备好了一段模板A存在了内存中,并以此为基础来打出文章B.盾盾每次操作可以将内存中 ...

  5. Python标准库之Sys模块使用详解

    sys 模块提供了许多函数和变量来处理 Python 运行时环境的不同部分. 处理命令行参数 在解释器启动后, argv 列表包含了传递给脚本的所有参数, 列表的第一个元素为脚本自身的名称. 使用sy ...

  6. xml语法、DTD约束xml、Schema约束xml、DOM解析xml

    今日大纲 1.什么是xml.xml的作用 2.xml的语法 3.DTD约束xml 4.Schema约束xml 5.DOM解析xml 1.什么是xml.xml的作用 1.1.xml介绍 在前面学习的ht ...

  7. Android 使用finalBitmap实现缓存读取

    public class NewsApplication extends Application{ private FinalBitmap finalBitmap=null; public Final ...

  8. 你未必知道的css小知识

    1:当按百分比设定一个元素的宽度时,它是相对于父容器的宽度计算的,但是,对于一些表示竖向距离的属性,例如padding-top,padding-bottom,margin-top,margin-bot ...

  9. iOS - UIToolbar

    前言 NS_CLASS_AVAILABLE_IOS(2_0) __TVOS_PROHIBITED @interface UIToolbar : UIView <UIBarPositioning& ...

  10. Win7_提高性能

    1. 设置成经典主题:桌面->右键->个性化->经典主题 2. 计算机->右键->属性->高级系统设置 ==> 系统属性->高级->性能-> ...