#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. 小扩展大用处,自己扩展一个ForeachRead吧

    是否用过IList的扩展方法 Foreach,而郁闷IEnumerable没有这个扩展?(没用过??用用吧,真的很方便,可以少好几行呢!!) 是否为了有一个索引而不得不用 for 而不能用 forea ...

  2. 我的android学习经历25

    android工程下R文件报错 今天我新建工程的时候,R文件报错,但是以前的文件并没有错误. 下面说一下我的情况: 我原来的工作区间是在D盘,后来我在E盘新建了一个工作区间,并且用E新建的工作区间,只 ...

  3. CF 321B Ciel and Duel(费用流)

    题目链接:http://codeforces.com/problemset/problem/321/B 题意:两个人,分别有n.m张牌.每张牌有两个属性类型和能力,类型为攻击或者防守.B的m张牌的属性 ...

  4. SignalR记录

    服务端检索数据库,有跟新,推送给客户端 1: GlobalHost.ConnectionManager.GetHubContext<tvHub>().Clients.Client(Clie ...

  5. __declspec关键字详细用法

    __declspec关键字详细用法 __declspec用于指定所给定类型的实例的与Microsoft相关的存储方式.其它的有关存储方式的修饰符如static与extern等是C和C++语言的ANSI ...

  6. mybatis的基本配置:实体类、配置文件、映射文件、工具类 、mapper接口

    搭建项目 一:lib(关于框架的jar包和数据库驱动的jar包) 1,第一步:先把mybatis的核心类库放进lib里

  7. HDU 3496 Watch The Movie(看电影)

    HDU 3496 Watch The Movie(看电影) Time Limit: 1000MS   Memory Limit: 65536K [Description] [题目描述] New sem ...

  8. python_way ,day2 字符串,列表,字典,时间模块

    python_way ,day2 字符串,列表,字典,自学时间模块 1.input: 2.0 3.0 区别 2.0中 如果要要用户交互输入字符串: name=raw_input() 如果 name=i ...

  9. 关于Docker 常用命令

    Docker 常用命令 分类列一下常用的CLI命令 仓库相关 search/ pull / push / login etc. 例:docker pull ubuntu 从仓库下载ubuntuimag ...

  10. C#正则表达式编程(一):C#中有关正则的类

    正则表达式是一门灵活性非常强的语言,匹配同样的字符串可能在不同的开发人员那里会得到不同的结果,在平常的时候也是用的时候看看相关资料,不用的时候就丢在脑后了,尽管在处理大部分情况下都能迅速处理,但是处理 ...