HYSBZ 2243 染色 (树链拆分)

HYSBZ 2243 染色 (树链拆分)

大家好,又见面了,我是全栈君,今天给大家准备了Idea注册码。

主题链接~~>

做题情绪:这题思路好想。调试代码调试了好久。第一次写线段树区间合并。

解题思路:

             树链剖分 + 线段树区间合并

       线段树的端点记录左右区间的颜色。颜色数目。合并的时候就用区间合并的思想。

还要注意一点。在由一条链转到还有一条链的时候要推断当前节点是否与父亲节点是否同一种颜色。

代码:

#include<iostream>
#include<sstream>
#include<map>
#include<cmath>
#include<fstream>
#include<queue>
#include<vector>
#include<sstream>
#include<cstring>
#include<cstdio>
#include<stack>
#include<bitset>
#include<ctime>
#include<string>
#include<cctype>
#include<iomanip>
#include<algorithm>
using namespace std  ;
#define INT __int64
#define L(x)  (x * 2)
#define R(x)  (x * 2 + 1)
const int INF = 0x3f3f3f3f ;
const double esp = 0.0000000001 ;
const double PI = acos(-1.0) ;
const int mod = 1e9 + 7 ;
const int MY = 1400 + 5 ;
const int MX = 100000 + 5 ;
int n ,m ,idx ,num ;
int head[MX] ,ti[MX] ,top[MX] ,dep[MX] ,siz[MX] ,son[MX] ,father[MX] ,g[MX] ;
struct Edge
{
    int v ,next ;
}E[MX*2] ;
void addedge(int u ,int v)
{
    E[num].v = v ; E[num].next = head[u] ; head[u] = num++ ;
    E[num].v = u ; E[num].next = head[v] ; head[v] = num++ ;
}
void dfs_find(int u ,int fa)
{
    dep[u] = dep[fa] + 1 ;
    siz[u] = 1 ;
    son[u] = 0 ;
    father[u] = fa ;
    for(int i = head[u] ;i != -1 ;i = E[i].next)
    {
        int v = E[i].v ;
        if(v == fa)  continue ;
        dfs_find(v ,u) ;
        siz[u] += siz[v] ;
        if(siz[son[u]] < siz[v])  son[u] = v ;
    }
}
void dfs_time(int u ,int fa)
{
    ti[u] = idx++ ;
    top[u] = fa ;
    if(son[u])  dfs_time(son[u] ,top[u]) ;
    for(int i = head[u] ;i != -1 ;i = E[i].next)
    {
        int v = E[i].v ;
        if(v == father[u] || v == son[u])   continue ;
        dfs_time(v ,v) ;
    }
}
struct node
{
    int le ,rt ,lc ,rc ,num ,add ;
}T[MX*4] ;
void build(int x ,int le ,int rt)
{
    T[x].le = le ; T[x].rt = rt ;
    T[x].lc = T[x].rc = T[x].add = -1 ;
    T[x].num = 0 ;
    if(le == rt)  return ;
    int Mid = (le + rt)>>1 ;
    build(L(x) ,le ,Mid) ;
    build(R(x) ,Mid + 1 ,rt) ;
}
void push_down(int x)
{
    if(T[x].add != -1)
    {
        // 左
        T[L(x)].lc = T[L(x)].rc = T[L(x)].add = T[x].add ; T[L(x)].num = 1 ;
        // 右
        T[R(x)].lc = T[R(x)].rc = T[R(x)].add = T[x].add ; T[R(x)].num = 1 ;
        T[x].add = -1 ;
    }
}
void push_up(int x)
{
    T[x].num = T[L(x)].num + T[R(x)].num ;
    if(T[L(x)].rc == T[R(x)].lc)
            T[x].num -- ;
    T[x].lc = T[L(x)].lc ; T[x].rc = T[R(x)].rc ;
}
void update(int x ,int le ,int rt ,int w)  // 更新某个区间
{
    if(T[x].le == le && T[x].rt == rt)
    {
        T[x].add = w ;
        T[x].lc = w ; T[x].rc = w ; T[x].num = 1 ;
        return ;
    }
    push_down(x) ;
    int Mid = (T[x].le + T[x].rt)>>1 ;
    if(le > Mid)    update(R(x) ,le ,rt ,w) ;
    else if(rt <= Mid)   update(L(x) ,le ,rt ,w) ;
    else
    {
        update(L(x) ,le ,Mid ,w) ;
        update(R(x) ,Mid+1 ,rt ,w) ;
    }
    push_up(x) ;
}
int Query(int x ,int le ,int rt)
{
    if(T[x].le == le && T[x].rt == rt)
        return T[x].num ;
    int Mid = (T[x].le + T[x].rt)>>1 ;
    push_down(x) ;
    if(le > Mid)   return   Query(R(x) ,le ,rt) ;
    else if(rt <= Mid) return   Query(L(x) ,le ,rt) ;
    else
    {
        int mx = 0 ;
        if(T[L(x)].rc == T[R(x)].lc)
             mx = -1 ;
        return  Query(L(x) ,le ,Mid) + Query(R(x) ,Mid+1 ,rt) + mx ;
    }
    push_up(x) ;
}
int Querynode(int x ,int pos)
{
    if(T[x].le == T[x].rt)
        return T[x].lc ;
    int Mid = (T[x].le + T[x].rt)>>1 ;
    push_down(x) ;
    if(pos <= Mid)
           return   Querynode(L(x) ,pos) ;
    else   return   Querynode(R(x) ,pos) ;
    push_up(x) ;
}
void LCAC(int u ,int v ,int w)
{
    while(top[u] != top[v])
    {
        if(dep[top[u]] < dep[top[v]])
             swap(u ,v) ;
        update(1 ,ti[top[u]] ,ti[u] ,w) ;
        u = father[top[u]] ;
    }
    if(dep[u] > dep[v])
         swap(u ,v) ;
    update(1 ,ti[u] ,ti[v] ,w) ;
}
int LCAQ(int u ,int v)
{
    int ans = 0 ;
    while(top[u] != top[v])
    {
        if(dep[top[u]] < dep[top[v]])
             swap(u ,v) ;
        ans += Query(1 ,ti[top[u]] ,ti[u]) ;
        if(Querynode(1 ,ti[top[u]]) == Querynode(1 ,ti[father[top[u]]]))
                      ans-- ;
        u = father[top[u]] ;
    }
    if(dep[u] > dep[v])
        swap(u ,v) ;
    ans += Query(1 ,ti[u] ,ti[v]) ;
    return  ans ;
}
int main()
{
    while(~scanf("%d%d" ,&n ,&m))
    {
        int u ,v ,w ;
        num = 0 ;
        memset(head ,-1 ,sizeof(head)) ;
        for(int i = 1 ;i <= n ; ++i)
            scanf("%d" ,&g[i]) ;
        for(int i = 1 ;i < n ; ++i)
        {
            scanf("%d%d" ,&u ,&v) ;
            addedge(u ,v) ;
        }
        dep[1] = siz[0] = 0 ;
        dfs_find(1 ,1) ;
        idx = 1 ;
        dfs_time(1 ,1) ;
        build(1 ,1 ,n) ;
        for(int i = 1 ;i <= n ; ++i)
            update(1 ,ti[i] ,ti[i] ,g[i]) ;
        char s[5] ;
        for(int i = 0 ;i < m ; ++i)
        {
            scanf("%s" ,s) ;
            scanf("%d%d" ,&u ,&v) ;
            if(s[0] == 'C')
            {
                scanf("%d" ,&w) ;
                LCAC(u ,v ,w) ;
            }
            else if(s[0] == 'Q')
                  printf("%d\n" ,LCAQ(u ,v)) ;
        }
    }
    return 0 ;
}

版权声明:本文博客原创文章,博客,未经同意,不得转载。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/117566.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 基于IP路由的GSLB《CDN技术详解》

    基于IP路由的GSLB《CDN技术详解》基于IP路由的GSLB是基于路由器原有的路由算法和数据包转发能力工作的。如图所示,有两个本地均衡器1和2,放在不同的POP点中,负责各自POP点内的服务器的负载均衡。先为这两个本地均衡器配置一个相同的VIP地址,对IP网上的路由器来说,这是到同一个IP地址的两条不同的路由。1)当终端a输入URL访问网站时,DNS系统会把VIP作为域名解析结果反馈给终端。2)终端向这个VIP发送请求时,请求数…

    2022年5月22日
    46
  • Automation服务器不能创建对象(金税盘)

    Automation服务器不能创建对象(金税盘)  1.安装防伪开票组件接口软件;  2.把接口的注册文件放到%防伪开票系统的安装目录%\BIN下  3.把%防伪开票系统的安装目录%\bin下的TaxCardX.dll文件复制至c:\windows\system32下  4.注册dll:在开始菜单中选择”运行”,输入”regsvr32TaxCardX.dll”5.  备注:此处电脑可能会提示不兼容,如:  …

    2022年6月1日
    35
  • jenkins教程菜鸟_jenkins新手入门教程

    jenkins教程菜鸟_jenkins新手入门教程Jenkins是一个开源软件项目,是基于Java开发的一种持续集成工具,用于监控持续重复的工作,旨在提供一个开放易用的软件平台,使软件的持续集成变成可能。虽然jenkins提供了Window、Linux、OSX等安装程序,但是都没有war方便。本文用的是jenkins的war包,简单、方便、快捷。一、安装jenkins1.在https://jenkins.io/下载jenkins.war;##…

    2022年5月14日
    46
  • 深度揭秘垃圾回收底层,这次让你彻底弄懂她

    深度揭秘垃圾回收底层,这次让你彻底弄懂她

    2020年11月20日
    159
  • “备份集中的数据库备份与现有的数据库不同”解决方法「建议收藏」

    “备份集中的数据库备份与现有的数据库不同”解决方法「建议收藏」原文发布时间为:2010-09-16——来源于本人的百度文章[由搬家工具导入]最主要就是要在“选项”中选择“覆盖现有数据库”,否则就会出现“备份集中的数据库备份与现有的数据库”的问题。以前一直使用SQLServer2000,现在跟潮流都这么紧,而且制定要求使用SQLServer2005,就在现在的项目中使用它了。对于SQLServer2005,有几个地方是要注意的,比方在还原数据库…

    2022年4月30日
    50
  • js判断是否是数字——isNaN()函数

    js判断是否是数字——isNaN()函数isNaN  返回一个Boolean值,指明提供的值是否是保留值NaN(不是数字)。  NaN即NotaNumber  isNaN(numValue)  必选项numvalue参数为要检查是否为NAN的值。  说明  假如值是NaN,那么isNaN函数返回true,否则返回false。使用这个函数的典型情况是检查parseInt

    2022年6月18日
    51

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号