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 ;
}

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

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

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

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • faster-rcnn 之 RPN网络的结构解析

    faster-rcnn 之 RPN网络的结构解析【说明】:我想很多人在看faster-rcnn的时候,都会被RPN的网络结构和连接方式纠结,作者在文中说的不是很清晰,这里给出解析;【首先】:大家应该要了解卷积神经网络的连接方式,卷积核的维度,反向传播时是如何灵活的插入一层,这些要了解;这里我推荐一份资料,真是写的非常清晰,就是MatConvet的用户手册,这个框架底层借用的是caffe的算法,所以数据结构,

  • ubuntu解压命令

    ubuntu解压命令本文转自:https://www.jb51.net/article/112207.htm 本文主要介绍的是在在Ubuntu下同下各种压缩与解压的方式,压缩与解压的重要性这里就不多说,我们下面直接来看详细的介绍吧。一、.tar解包:tarxvfFileName.tar打包:tarcvfFileName.tarDirName(注:tar是打包,不是压缩!)二、.g…

  • 数据结构:树

    数据结构:树

  • Apache配置虚拟主机出现forbidden的问题

    Apache配置虚拟主机出现forbidden的问题1.httpd.conf中配置过网站根目录该配置文件的250行左右,修改了根目录①httpd.conf配置文件中这两处都要修改②httpd-vhosts.conf配置文件中检查DocumentRoot是不是有默认值,有的话要修改根目录③重启Apache服务,清空浏览器缓存2.httpd-vhosts.conf中配置过虚拟主机httpd.conf配置中文件中找到VirtualHost,…

  • 简单了解Activity工作流引擎

    简单了解Activity工作流引擎一、什么是工作流以请假为例,现在大多数公司的请假流程是这样的员工打电话(或网聊)向上级提出请假申请——上级口头同意——上级将请假记录下来——月底将请假记录上交公司——公司将请假录入电脑采用工作流技术的公司的请假流程是这样的员工使用账户登录系统——点击请假——上级登录系统点击允许就这样,一个请假流程就结束了有人会问,那上级不用向公司提交请假记录?公司不用将记录录入电脑?答案…

  • Python将数据写入txt文件_python将内容写入txt文件

    Python将数据写入txt文件_python将内容写入txt文件一、读写txt文件1、打开txt文件Note=open(‘x.txt’,mode=’w’)函数=open(x.扩展名,mode=模式)模式种类:w只能操作写入(如果而文件中有数据,再次写入内容,会把原来的覆盖掉)r只能读取a向文件追加w+可读可写r+可读可写a+可读可追加wb+写入进制数据2、向文件中写入数据第一种写入方式:write写入Note.write…

发表回复

您的电子邮箱地址不会被公开。

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