【题意】给定n个点的树,每条边有一个小写字母a~v,求每棵子树内的最长回文路径,回文路径定义为路径上所有字母存在一种排列为回文串。n<=5*10^5。

【算法】dsu on tree

【题解】这题经典套路就是按照22个字母个数的奇偶性压位,然后两段路径异或起来是0或1<<j就是合法路径。

dsu的时候每个点统计其子树内经过这个点的路径,注意包括从子树到该点终止的和该点自身也要算。

那么类似点分治的方式,算完重儿子后处理一下根,然后就一棵一棵轻儿子子树和之前的子树状态桶数组统计然后加入。

传递上去的时候需要特别注意,dsu是无法支持数组的整体位移的,解决方法一般是把统计从x到子树改为从根到子树,这样所有点都是一样的,不需要位移。

当然这就需要满足信息的可减性,而深度deep和异或xor都是满足的。(xor和deep的两点间路径转两点到根路径非常经典了)

复杂度O(n log n)。

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. bool isdigit(char c){return c>=''&&c<='';}
  5. int read(){
  6. int s=,t=;char c;
  7. while(!isdigit(c=getchar()))if(c=='-')t=-;
  8. do{s=s*+c-'';}while(isdigit(c=getchar()));
  9. return s*t;
  10. }
  11. using namespace std;
  12. const int maxn=,inf=0x3f3f3f3f;
  13. int n,sz[maxn],first[maxn],a[maxn],b[],c[],ans[maxn],w[maxn],fa[maxn],tot,deep[maxn];
  14. struct edge{int v,from;}e[maxn*];
  15. void insert(int u,int v){tot++;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
  16. void p(int &x,int y){if(x<y)x=y;}
  17. void dfs_pre(int x){
  18. sz[x]=;
  19. for(int i=first[x];i;i=e[i].from){
  20. deep[e[i].v]=deep[x]+;
  21. a[e[i].v]^=a[x];
  22. dfs_pre(e[i].v);
  23. sz[x]+=sz[e[i].v];
  24. if(sz[e[i].v]>sz[w[x]])w[x]=e[i].v;
  25. }
  26. }
  27. void calc(int x,int tp){
  28. for(int j=;j<=;j++)p(ans[tp],deep[x]+c[a[x]^b[j]]);
  29. for(int i=first[x];i;i=e[i].from)calc(e[i].v,tp);
  30. }
  31. void add(int x,int k){
  32. if(k)p(c[a[x]],deep[x]);
  33. else c[a[x]]=-inf;
  34. for(int i=first[x];i;i=e[i].from)add(e[i].v,k);
  35. }
  36. void dfs(int x){
  37. for(int i=first[x];i;i=e[i].from)if(e[i].v!=w[x])dfs(e[i].v);
  38. if(w[x])dfs(w[x]);//
  39. p(c[a[x]],deep[x]);for(int j=;j<=;j++)p(ans[x],deep[x]+c[a[x]^b[j]]);
  40. for(int i=first[x];i;i=e[i].from)if(e[i].v!=w[x])calc(e[i].v,x),add(e[i].v,);
  41. if(x!=w[fa[x]])add(x,);
  42. }
  43. char s[];
  44. int main(){
  45. n=read();
  46. for(int i=;i<=n;i++){
  47. fa[i]=read();if(fa[i])insert(fa[i],i);
  48. scanf("%s",s);a[i]=<<(s[]-'a');
  49. }
  50. b[]=;for(int j=;j<=;j++)b[j]=<<j;
  51. for(int i=;i<(<<);i++)c[i]=-inf;
  52. dfs_pre();dfs();
  53. for(int i=;i<=n;i++)ans[i]-=*deep[i];
  54. for(int i=n;i>=;i--)p(ans[fa[i]],ans[i]);
  55. for(int i=;i<=n;i++)printf("%d ",ans[i]);
  56. return ;
  57. }

即使信息满足可减性,dsu on tree也不能像点分治一样删除某棵子树信息,进去统计后再加回来。因为dsu on tree必须满足【不能遍历重儿子】,否则复杂度就会爆炸。

不过如果题目要求的是除了某棵子树外的信息,就可以做,先统计所有轻儿子做除了重儿子的,然后进重儿子后统计所有轻儿子,一个一个删除来做除了某个轻儿子的。

【CodeForces】741 D. Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree)的更多相关文章

  1. 【cf741】D. Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree)

    传送门 题意: 给出一颗以\(1\)为根的有根树,树边带有一个字符(\(a\)~\(v\))的信息. 输出对于每个结点,其子树内最长的简单路径并且满足边上的字符能够组成回文串. 思路: 显然最终的答案 ...

  2. 【CF741D】Arpa’s letter-marked tree and Mehrdad’s Dokhtar-kosh paths(dsu on tree)

    题意:我们称一个字符串为周驿东串当且仅当重排它的字符可以组成一个回文串. 给出一个n个点的有根树,根为1,每条边上有一个从a到v的字符,求每个点的子树中所有简单路径可以组成的周驿东串中的最长长度. n ...

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

    感觉dsu on tree一定程度上还是与点分类似的.考虑求出跨过每个点的最长满足要求的路径,再对子树内取max即可. 重排后可以变成回文串相当于出现奇数次的字母不超过1个.考虑dsu on tree ...

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

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

  5. 【Codeforces】Round #491 (Div. 2) 总结

    [Codeforces]Round #491 (Div. 2) 总结 这次尴尬了,D题fst,E没有做出来.... 不过还好,rating只掉了30,总体来说比较不稳,下次加油 A:If at fir ...

  6. 【Codeforces】Round #488 (Div. 2) 总结

    [Codeforces]Round #488 (Div. 2) 总结 比较僵硬的一场,还是手速不够,但是作为正式成为竞赛生的第一场比赛还是比较圆满的,起码没有FST,A掉ABCD,总排82,怒涨rat ...

  7. 【CodeForces】901 B. GCD of Polynomials

    [题目]B. GCD of Polynomials [题意]给定n,要求两个最高次项不超过n的多项式(第一个>第二个),使得到它们GCD的辗转次数为n.n<=150. [算法]构造 [题解 ...

  8. 【CF600E】Lomsat gelral(dsu on tree)

    [CF600E]Lomsat gelral(dsu on tree) 题面 洛谷 CF题面自己去找找吧. 题解 \(dsu\ on\ tree\)板子题 其实就是做子树询问的一个较快的方法. 对于子树 ...

  9. 【Silverlight】Bing Maps开发应用与技巧一:地图打点与坐标控件(CoordControl)

    [Silverlight]Bing Maps开发应用与技巧一:地图打点与坐标控件(CoordControl) 使用Bing Maps Silverlight Control开发中,很多时候都需要实现在 ...

随机推荐

  1. android自动化之appium的环境搭建

    简介appium     appium是C/S架构,appium的核心是一个web服务器,它提供了一套REST的接口,他会接收客户端的连接,监听到命令.执行会再将结果通过HTTP响应返还给客户端.ap ...

  2. jmeter 多线程组间变量共享

    jmeter的线程组之间是相互独立的,各个线程组互不影响,所以线程组A中输出的参数,是无法直接在线程组B中被调用的. 但是有时为了方便管理,我们可能是把各个接口单独存放在不同的线程组中.拿Cookie ...

  3. python web调用docker-py

    在 /etc/init.d/docker的start()函数末尾加入:chmod 777 /var/run/docker.sock 否则web程序会没有权限去操作  

  4. vector(char*)和vector(string)

    vector<char*> ch; vector<string> str; for(int i=0;i<5;i++) { char *c=fun1();//通过这个语句产 ...

  5. ySQL性能优化的21个最佳实践 和 mysql使用索引

    MySQL性能优化的21个最佳实践 和 mysql使用索引 今天,数据库的操作越来越成为整个应用的性能瓶颈了,这点对于Web应用尤其明显.关于数据库的性能,这并不只是DBA才需要担心的事,而这更是我 ...

  6. PGM学习之三 朴素贝叶斯分类器(Naive Bayes Classifier)

    介绍朴素贝叶斯分类器的文章已经很多了.本文的目的是通过基本概念和微小实例的复述,巩固对于朴素贝叶斯分类器的理解. 一 朴素贝叶斯分类器基础回顾 朴素贝叶斯分类器基于贝叶斯定义,特别适用于输入数据维数较 ...

  7. BZOJ5312 冒险(线段树)

    记录区间and/or,修改时如果对整个区间影响都相同就打标记,否则递归.复杂度不太会证. #include<iostream> #include<cstdio> #includ ...

  8. WEB入门.九 导航菜单

    学习内容 水平导航菜单 垂直导航菜单 下拉式导航菜单 能力目标 制作tab标签导航菜单 制作带箭头的导航菜单 制作带信息提示的导航菜单 制作垂直下拉导航菜单 制作水平下拉导航菜单 本章简介 上一章节中 ...

  9. mac 命令行大杂烩

    一.实用的 1.代替 cat 的工具:bat,支持语法高亮.同时显示行号,使用: bat xx.yyy 安装:brew install bat 2.man 命令的替代品:tldr 安装:brew in ...

  10. C++下实现同接口下多个类作为参数的调用和传参

    /* 实现同接口下不同类的对象的转移 定义类的接口 定义多个继承该接口的类 定义管理类,把接口当作类型, 传入该接口下各种类的对象,进行操作 */ #include<iostream> # ...