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](http://codeforces.com/contest/741/problem/D)
题意:
一棵有根树,边上有字母a~v,求每个子树中最长的边,满足这个边上的所有字母重拍后可以构成回文
发明者自己出的题...orz
由于本来知道就是dsu on tree,所以还是想出来了
首先点分治是没法做了,这是有根树
写成二进制,两条链合起来构成回文\(\rightarrow\)异或和为0或者只有一位是1
一开始困惑于只处理到当前根的异或和的话,随着当前根的变化异或值会变
然后发现我们可以处理到根的异或和,两条链异或之后\(lca\)之上的部分正好没有了
用\(f[i]\)记录到根的异或和为\(i\)的最大深度
进行dsu on tree
当前子树的答案先\(max\)一下孩子的答案,然后类似于点分治处理经过当前根的路径
和点分治一样,一个一个轻儿子处理,先更新答案再更新信息\(f\)
如果当前是父亲的轻儿子,那么要清空\(f\)的信息
说一下起初的错误:
- 单独用了一个\(g\)记录轻儿子,这是不对的,因为当前的子树要么是重儿子,应该一直保留;要么是轻儿子,到时候全清除就可以了。
- 一定不能整体操作重儿子的信息,要不然复杂度就不对了
- 这里有一个\(trick\),因为有根树处理一棵子树我们可以提前搞出\(dfs\)序就不用每次\(dfs\)了,常数优化显著
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
#define pii pair<int, int>
#define MP make_pair
#define fir first
#define sec second
const int N=5e5+5, M=1e7+5;
int read(){
char c=getchar();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1; c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0'; c=getchar();}
return x*f;
}
int n, x;
char s[5];
struct edge{int v, ne, c;}e[N<<1];
int cnt, h[N];
inline void ins(int u, int v, int c) {
e[++cnt]=(edge){v, h[u], c}; h[u]=cnt;
}
int Xor[N], dfc, ver[N];
pii dfn[N];
void dfsPre(int u) {
dfn[u].fir = ++dfc; ver[dfc] = u;
for(int i=h[u];i;i=e[i].ne) Xor[e[i].v] = Xor[u]^e[i].c, dfsPre(e[i].v);
dfn[u].sec = dfc;
}
int size[N], mx[N], deep[N];
void dfs(int u) {
size[u]=1;
for(int i=h[u];i;i=e[i].ne) {
deep[e[i].v] = deep[u]+1;
dfs(e[i].v);
size[u] += size[e[i].v];
if(size[e[i].v] > size[mx[u]]) mx[u] = e[i].v;
}
}
int ans[N], f[M];
void dfs(int u, int keep) {
int &cur = ans[u];
for(int i=h[u];i;i=e[i].ne)
if(e[i].v != mx[u]) dfs(e[i].v, 0), cur = max(cur, ans[e[i].v]);
if(mx[u]) dfs(mx[u], 1), cur = max(cur, ans[mx[u]]);
int now = Xor[u], d = deep[u];
if(f[now]) cur = max(cur, f[now] - d);
for(int i=0; i<=21; i++)
if(f[now^(1<<i)]) cur = max(cur, f[now^(1<<i)] - d);
f[now] = max(f[now], d);
for(int i=h[u];i;i=e[i].ne) if(e[i].v != mx[u]) {
int l = dfn[e[i].v].fir, r = dfn[e[i].v].sec;
for(int j=l; j<=r; j++) {
int x = ver[j], now = Xor[x];
if(f[now]) cur = max(cur, f[now] + deep[x] - (d<<1));
for(int i=0; i<=21; i++)
if(f[now^(1<<i)]) cur = max(cur, f[now^(1<<i)] + deep[x] - (d<<1));
}
for(int j=l; j<=r; j++)
f[ Xor[ver[j]] ] = max(f[ Xor[ver[j]] ], deep[ver[j]]);
}
if(!keep) {
int l = dfn[u].fir, r = dfn[u].sec;
for(int j=l; j<=r; j++) f[ Xor[ver[j]] ] = 0;
}
}
int main() {
//freopen("in","r",stdin);
n=read();
for(int i=2; i<=n; i++) x=read(), scanf("%s",s), ins(x, i, 1<<(s[0]-'a') );
dfsPre(1);
deep[1]=1; dfs(1);
dfs(1, 0);
for(int i=1; i<=n; i++) printf("%d ",ans[i]);
}
这里是没用dfs序的版本
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
#define pii pair<int, int>
#define MP make_pair
#define fir first
#define sec second
const int N=5e5+5, M=1e7+5;
int read(){
char c=getchar();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1; c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0'; c=getchar();}
return x*f;
}
int n, x;
char s[5];
struct edge{int v, ne, c;}e[N<<1];
int cnt, h[N];
inline void ins(int u, int v, int c) { //printf("ins %d %d %d\n",u,v,c);
e[++cnt]=(edge){v, h[u], c}; h[u]=cnt;
}
int Xor[N];
void dfsXor(int u) {
for(int i=h[u];i;i=e[i].ne) Xor[e[i].v] = Xor[u]^e[i].c, dfsXor(e[i].v);
}
int size[N], mx[N], deep[N], big[N];
void dfs(int u) {
size[u]=1;
for(int i=h[u];i;i=e[i].ne) {
deep[e[i].v] = deep[u]+1;
dfs(e[i].v);
size[u] += size[e[i].v];
if(size[e[i].v] > size[mx[u]]) mx[u] = e[i].v;
}
}
int ans[N], f[M];
void update(int u, int &ans, int d) {
int now = Xor[u];
if(f[now]) ans = max(ans, f[now] + deep[u] - 2*d);
for(int i=0; i<=21; i++) {
int t = now^(1<<i);
if(f[t]) ans = max(ans, f[t] + deep[u] - 2*d);
}
for(int i=h[u];i;i=e[i].ne) if(!big[e[i].v]) update(e[i].v, ans, d);
}
void merge(int u) {
int now = Xor[u];
f[now] = max(f[now], deep[u]);
for(int i=h[u];i;i=e[i].ne) if(!big[e[i].v]) merge(e[i].v);
}
void clear(int u) {
int now = Xor[u];
f[now] = 0;
for(int i=h[u];i;i=e[i].ne) if(!big[e[i].v]) clear(e[i].v);
}
void dfs(int u, int keep) { //printf("dfs %d %d mx %d\n",u,keep,mx[u]);
for(int i=h[u];i;i=e[i].ne)
if(e[i].v != mx[u]) dfs(e[i].v, 0), ans[u] = max(ans[u], ans[e[i].v]);
if(mx[u]) dfs(mx[u], 1), big[mx[u]]=1, ans[u] = max(ans[u], ans[mx[u]]);
int now = Xor[u], d = deep[u];
if(f[now]) ans[u] = max(ans[u], f[now] + deep[u] - 2*d);
for(int i=0; i<=21; i++) {
int t = now^(1<<i);
if(f[t]) ans[u] = max(ans[u], f[t] + deep[u] - 2*d);
}
f[now] = max(f[now], d);
for(int i=h[u];i;i=e[i].ne)
if(e[i].v != mx[u]) update(e[i].v, ans[u], deep[u]), merge(e[i].v);
big[mx[u]]=0;
if(!keep) clear(u);
}
int main() {
//freopen("in","r",stdin);
n=read();
for(int i=2; i<=n; i++) x=read(), scanf("%s",s), ins(x, i, 1<<(s[0]-'a') );
dfsXor(1);
//puts("");for(int i=1; i<=n; i++) printf("Xor %d %d\n",i,Xor[i]);puts("");
deep[1]=1; dfs(1);
dfs(1, 0);
for(int i=1; i<=n; i++) printf("%d ",ans[i]);
}
CF 741D. Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths [dsu on tree 类似点分治]的更多相关文章
- Codeforces.741D.Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree 思路)
题目链接 \(Description\) 给定一棵树,每条边上有一个字符(a~v).对每个节点,求它的子树中一条最长的路径,满足 路径上所有边上的字符可以重新排列成一个回文串.输出其最长长度. \(n ...
- [Codeforces741D]Arpa's letter-marked tree and Mehrdad's Dokhtar-kosh paths——dsu on tree
题目链接: Codeforces741D 题目大意:给出一棵树,根为$1$,每条边有一个$a-v$的小写字母,求每个点子树中的一条最长的简单路径使得这条路径上的边上的字母重排后是一个回文串. 显然如果 ...
- CF741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths——dsu on tree
题目描述 一棵根为1 的树,每条边上有一个字符(a-v共22种). 一条简单路径被称为Dokhtar-kosh当且仅当路径上的字符经过重新排序后可以变成一个回文串. 求每个子树中最长的Dokhtar- ...
- CF741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths (dsu on tree) 题解
先说一下dsu算法. 例题:子树众数问题. 给出一棵树,每个点有点权,求每个子树中出现次数最多的数的出现次数. 树的节点数为n,\(n \leq 500000\) 这个数据范围,\(O(n \sqrt ...
- 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 ...
- [探究] dsu on tree,一类树上离线问题的做法
dsu on tree. \(\rm 0x01\) 前言\(\&\)技术分析 \(\bold{dsu~on~tree}\),中文别称"树上启发式合并"(虽然我并不承认这种称 ...
- dsu on tree (树上启发式合并) 详解
一直都没出过算法详解,昨天心血来潮想写一篇,于是 dsu on tree 它来了 1.前置技能 1.链式前向星(vector 建图) 2.dfs 建树 3.剖分轻重链,轻重儿子 重儿子 一个结点的所有 ...
- codeforces 741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(启发式合并)
codeforces 741D Arpa's letter-marked tree and Mehrdad's Dokhtar-kosh paths 题意 给出一棵树,每条边上有一个字符,字符集大小只 ...
- Codeforces 741D Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree)
感觉dsu on tree一定程度上还是与点分类似的.考虑求出跨过每个点的最长满足要求的路径,再对子树内取max即可. 重排后可以变成回文串相当于出现奇数次的字母不超过1个.考虑dsu on tree ...
随机推荐
- 使用Eclipse在Excel中找出两张表中相同证件号而姓名或工号却出现不同的的项
1:首先把Excel中的文本复制到txt中,复制如下: A表: 证件号 工号 姓名 310110xxxx220130004 101 傅家宜3101 ...
- Linux中安装opencv-3.3.1
在ubuntu16.04中安装opencv3.3.1的过程中踩了许多坑.一开始直接安装还挺顺利但运行程序时总是提示libgtk2.0-dev和pkg-config没有安装,在安装这两个包的过程中也不顺 ...
- Spark算子--mapValues
转载请标明出处http://www.cnblogs.com/haozhengfei/p/ccc9d6b5c46ac7209c1e104bd219bfb4.html mapValues--Transfo ...
- 基于VUE选择上传图片并在页面显示(图片可删除)
demo例子: 依赖文件 : http://files.cnblogs.com/files/zhengweijie/jquery.form.rar HTML文本内容: <template> ...
- WIN2016安装织梦没写入权限怎么办听语音
配置好了WINSERVER2016环境,一切看起来都弄得差不多了,可是安装织梦的时候提示我没有写入权限,不能继续安装,于是我很郁闷,开始寻求解决办法. 工具/原料 WINSERVER2016 织梦5. ...
- 修改国内yum源
yum的源配置文件名为:CentOS-Base.repo 一般情况是在/etc/yum.repos.d目录下有CentOS-Base.repo 备份CentOS-Base.repo文件 打开这个网站h ...
- WaitForXXX等待无效句柄
=================================版权声明================================= 版权声明:原创文章 禁止转载 请通过右侧公告中的“联系邮 ...
- Python 魔法方法详解
据说,Python 的对象天生拥有一些神奇的方法,它们总被双下划线所包围,他们是面向对象的 Python 的一切. 他们是可以给你的类增加魔力的特殊方法,如果你的对象实现(重载)了这些方法中的某一个, ...
- Effective Java 第三版——28. 列表优于数组
Tips <Effective Java, Third Edition>一书英文版已经出版,这本书的第二版想必很多人都读过,号称Java四大名著之一,不过第二版2009年出版,到现在已经将 ...
- ASP.net core 2.0.0 中 asp.net identity 2.0.0 的基本使用(四)—用户注册
修改用户注册 1.修改用户名注册规则. 打开Controllers目录下的AccountController.cs. 在控制器中找到 public async Task<IActionResul ...