题目描述

一棵根为1 的树,每条边上有一个字符(a-v共22种)。 一条简单路径被称为Dokhtar-kosh当且仅当路径上的字符经过重新排序后可以变成一个回文串。 求每个子树中最长的Dokhtar-kosh路径的长度。

思路

  不多的只能用树上启发式合并做的题。。。虽然算法很暴力,但是本题还是挺难想的。

  考虑一共只有22种不同的边权,我们要找的是经过重新排序后回文的路径,也就是说:为偶数时,必须有两两相同的;为奇数时,最多只能有一个多出来的。那么我们考虑状压,对第$i$种边权压成$1<<i-1$,这样我们找的路径就变成了:路径上所有边的异或和为0或者为22中状态的一种。(为什么要状压呢?随便举几个栗子就知道了)

  我们先处理出$d[i]表示i到根的异或和,那么任意路径(u,v)的异或和就是d[u] \oplus d[v],暴力统计时,我们先遍历一棵轻儿子,遍历完后再把轻儿子的贡献加入桶中,这样就可以做到让u变成此次暴力的lca,于是我们在now结点的子树中遍历到(u,v)时,先统计桶中有没有d[u],这样异或起来为0,再统计有没有和d[u]异或起来为2^i$的即可。

code

#include<bits/stdc++.h>
#define I inline
using namespace std;
const int N=;
const int inf=(<<)-;
int val[N],n;
struct node
{
int to,nxt,w;
}g[N];
int head[N],cnt; int d[N],sz[N],son[N],Son,buk[<<],now,dep[N],ans[N]; I int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
} I void addedge(int u,int v,int w)
{
g[++cnt].nxt=head[u];
g[cnt].to=v;
g[cnt].w=w;
head[u]=cnt;
} I void get_son(int u)
{
sz[u]=;
for(int i=head[u];i;i=g[i].nxt)
{
int v=g[i].to,w=g[i].w;
d[v]=d[u]^w;dep[v]=dep[u]+;
get_son(v);
sz[u]+=sz[v];
if(sz[v]>sz[son[u]])son[u]=v;
}
} I void init(int u)
{
buk[d[u]]=-inf;
for(int i=head[u];i;i=g[i].nxt)init(g[i].to);
} I void get(int u)
{
ans[now]=max(ans[now],dep[u]+buk[d[u]]);
for(int i=;i<=;i++)ans[now]=max(ans[now],dep[u]+buk[(<<i)^d[u]]);
for(int i=head[u];i;i=g[i].nxt)
{
int v=g[i].to;get(v);
}
} I void add(int u)
{
buk[d[u]]=max(buk[d[u]],dep[u]);
for(int i=head[u];i;i=g[i].nxt)
{
int v=g[i].to;add(v);
}
} I void dfs(int u,bool op)
{
for(int i=head[u];i;i=g[i].nxt)
{
int v=g[i].to;
if(v==son[u])continue;
dfs(v,);
}
if(son[u])dfs(son[u],),Son=son[u];
now=u;
for(int i=head[u];i;i=g[i].nxt)
{
int v=g[i].to;
if(v==Son)continue;
get(v);add(v);
}
buk[d[u]]=max(buk[d[u]],dep[u]);
ans[u]=max(ans[u],buk[d[u]]+dep[u]);
for(int i=;i<=;i++)ans[u]=max(ans[u],dep[u]+buk[(<<i)^d[u]]);
ans[u]-=dep[u]<<;
for(int i=head[u];i;i=g[i].nxt)
{
int v=g[i].to;
ans[u]=max(ans[u],ans[v]);
}
if(!op)init(u);
} int main()
{
n=read();
memset(buk,,sizeof(buk));
for(int i=;i<=n;i++)
{
int x=read();
char ch=getchar();
while(ch<'a'||ch>'v')ch=getchar();
addedge(x,i,(<<(ch-'a')));
d[i]=<<(ch-'a');
}
get_son();
dfs(,);
for(int i=;i<=n;i++)printf("%d ",ans[i]);
}

CF741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths——dsu on tree的更多相关文章

  1. CF 741D. Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths [dsu on tree 类似点分治]

    D. Arpa's letter-marked tree and Mehrdad's Dokhtar-kosh paths CF741D 题意: 一棵有根树,边上有字母a~v,求每个子树中最长的边,满 ...

  2. [Codeforces741D]Arpa's letter-marked tree and Mehrdad's Dokhtar-kosh paths——dsu on tree

    题目链接: Codeforces741D 题目大意:给出一棵树,根为$1$,每条边有一个$a-v$的小写字母,求每个点子树中的一条最长的简单路径使得这条路径上的边上的字母重排后是一个回文串. 显然如果 ...

  3. Codeforces.741D.Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree 思路)

    题目链接 \(Description\) 给定一棵树,每条边上有一个字符(a~v).对每个节点,求它的子树中一条最长的路径,满足 路径上所有边上的字符可以重新排列成一个回文串.输出其最长长度. \(n ...

  4. CF741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths (dsu on tree) 题解

    先说一下dsu算法. 例题:子树众数问题. 给出一棵树,每个点有点权,求每个子树中出现次数最多的数的出现次数. 树的节点数为n,\(n \leq 500000\) 这个数据范围,\(O(n \sqrt ...

  5. codeforces 741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths

    题目链接:Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths 第一次写\(dsu\ on\ tree\),来记录一下 \(dsu\ o ...

  6. dsu on tree (树上启发式合并) 详解

    一直都没出过算法详解,昨天心血来潮想写一篇,于是 dsu on tree 它来了 1.前置技能 1.链式前向星(vector 建图) 2.dfs 建树 3.剖分轻重链,轻重儿子 重儿子 一个结点的所有 ...

  7. [探究] dsu on tree,一类树上离线问题的做法

    dsu on tree. \(\rm 0x01\) 前言\(\&\)技术分析 \(\bold{dsu~on~tree}\),中文别称"树上启发式合并"(虽然我并不承认这种称 ...

  8. CF741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths

    CF741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths 好像这个题只能Dsu On Tree? 有根树点分治 统计子树过x的 ...

  9. 【DSU on tree】【CF741D】Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths

    Description 给定一棵 \(n\) 个节点的树,每条边上有一个字符,字符集大小 \(22\),求每个节点的子树内最长的简单路径使得路径上的字符经过重排后构成回文串. Limitation \ ...

随机推荐

  1. 创建SSM项目所需

    一.mybatis所需: 1.相关jar包 2.创数据库+Javabean类 3.接口+写SQL的xml映射文件 4.核心配置文件:SqlMapConfig.xml 二.springMVC所需: 1. ...

  2. 安卓控件 仪表盘控件 柱状图控件 曲线控件 xamarin.android 分类器 瓶子控件 报警控件 水箱控件 进度条控件等

    本篇博客主要介绍一个控件库,HslControls.dll 的界面,这个控件库支持winform,winform的参考另一篇文章:https://www.cnblogs.com/dathlin/p/1 ...

  3. django-搭建BBS关键点总结

    0826自我总结 django-搭建BBS关键点总结 一.关于开口子,直接输入url访问文件内容 django自带开了个口子是static文件可以直接访问到 手动开口子 urs.py from dja ...

  4. cobalt strike笔记-常用beacon扫盲

    最近还是重新补一下cs的东西 0x01 Beacon命令 Beacon Commands =============== Command Description ------- ----------- ...

  5. POWERSPLOIT-Exfiltration(信息收集)脚本渗透实战

    Exfiltration模块 a) 调用Get-Keystrokes记录用户的键盘输入. 1)通过IEX下载并调用Get-Keystrokes. PS C:\Users\Administrator&g ...

  6. 数据挖掘:python数据清洗cvs里面带中文字符

    数据清洗,使用python数据清洗cvs里面带中文字符,意图是用字典对应中文字符,即key值是中文字符,value值是index,自增即可:利用字典数据结构没有重复key值的特性,把中文字符映射到了数 ...

  7. Ubuntu 重装vmtool

    1. 虚拟机菜单 ->  更新虚拟机  : 2. 弹出的窗口中: 3. 拷贝红色的文件到可读写的目录: 4. 解压,运行解压出来的绿色脚本文件,一路回车:

  8. go-json类

    package main import ( "encoding/json" "fmt" ) /* { "company":"itc ...

  9. pytest2-收集与执行测试用例规则

    pytest收集测试用例规则 测试文件以test_开头(以_test结尾也可以) 测试类以Test开头,并且不能带有 init 方法 测试函数以test_开头(以_test结尾也可以) pytest执 ...

  10. art-template的循环操作

    art-template的官方文档(网址:https://aui.github.io/art-template/docs/)中对循环的介绍如下: 循环 标准语法 {{each target}} {{$ ...