【题目】E. Palindromes in a Tree

【题意】给定一棵树,每个点都有一个a~t的字符,一条路径回文定义为路径上的字符存在一个排列构成回文串,求经过每个点的回文路径数。n<=2*10^5。

【算法】点分治

【题解】状压20位的二进制表示一条路径的字符状态,点分治过程中维护扫描过的路径只须维护状态桶数组,t[i]表示前面状态为i的路径条数。

合并:考虑当前状态为j,要使合并的状态满足条件即i^j=1<<k(0<=k<20)或i^j=0,移项得i=j^(1<<k)或i=j,所以路径数是Σ t [ j^(1<<k) ]+t[j]。

统计每个点:对于当前要处理的重心x,先遍历所有子树得到整个t[]数组,然后对每个子树先删除其在桶里的状态,然后扫一遍贡献子树中每个点,最后将子树的状态加回桶中。

这样可以做到每条路径都贡献到每个点,要特殊处理重心的贡献。

复杂度O(n log n)。

  1. #include<cstdio>
  2. #include<cctype>
  3. #include<algorithm>
  4. #define ll long long
  5. using namespace std;
  6. const int maxn=,maxN=;
  7. int tot,first[maxn],sz[maxn],vis[maxn],sum,root,a[maxn],u,v,n;
  8. ll ans[maxn],t[maxN];
  9. struct edge{int v,from;}e[maxn*];
  10.  
  11. void insert(int u,int v){tot++;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
  12. void getroot(int x,int fa){
  13. sz[x]=;
  14. bool ok=;
  15. for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa&&!vis[e[i].v]){
  16. getroot(e[i].v,x);
  17. sz[x]+=sz[e[i].v];
  18. if(sz[e[i].v]>sum/)ok=;
  19. }
  20. if(ok&&sz[x]>=sum/)root=x;
  21. }
  22. void dfs(int x,int fa,int p,int s){
  23. t[s^=(<<a[x])]+=p;
  24. for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa&&!vis[e[i].v])dfs(e[i].v,x,p,s);
  25. }
  26. ll calc(int x,int fa,int s){
  27. s^=(<<a[x]);ll num=t[s];
  28. for(int i=;i<;i++)num+=t[s^(<<i)];
  29. for(int i=first[x];i;i=e[i].from)if(e[i].v!=fa&&!vis[e[i].v])num+=calc(e[i].v,x,s);
  30. ans[x]+=num;
  31. return num;
  32. }
  33. void solve(int x,int s){
  34. vis[x]=;
  35. dfs(x,,,);
  36. ll num=t[];
  37. for(int i=;i<;i++)num+=t[<<i];
  38. for(int i=first[x];i;i=e[i].from)if(!vis[e[i].v]){
  39. dfs(e[i].v,x,-,<<a[x]);
  40. num+=calc(e[i].v,x,);
  41. dfs(e[i].v,x,,<<a[x]);
  42. }
  43. ans[x]+=num/;
  44. dfs(x,,-,);
  45. for(int i=first[x];i;i=e[i].from)if(!vis[e[i].v]){
  46. if(sz[e[i].v]>sz[x])sum=s-sz[x];else sum=sz[e[i].v];
  47. getroot(e[i].v,x);
  48. solve(root,sum);
  49. }
  50. }
  51. char s[maxn];
  52. int main(){
  53. scanf("%d",&n);
  54. for(int i=;i<n;i++){
  55. scanf("%d%d",&u,&v);
  56. insert(u,v);insert(v,u);
  57. }
  58. scanf("%s",s+);
  59. for(int i=;i<=n;i++)a[i]=s[i]-'a';
  60. sum=n;
  61. getroot(,);
  62. solve(root,sum);
  63. for(int i=;i<=n;i++)printf("%lld ",ans[i]+);
  64. return ;
  65. }

【CodeForces】914 E. Palindromes in a Tree 点分治的更多相关文章

  1. Codeforces 1039D You Are Given a Tree [根号分治,整体二分,贪心]

    洛谷 Codeforces 根号分治真是妙啊. 思路 考虑对于单独的一个\(k\)如何计算答案. 与"赛道修建"非常相似,但那题要求边,这题要求点,所以更加简单. 在每一个点贪心地 ...

  2. CF914E Palindromes in a Tree(点分治)

    link 题目大意:给定一个n个点的树,每个点都有一个字符(a-t,20个字符) 我们称一个路径是神犇的,当这个路径上所有点的字母的某个排列是回文 求出对于每个点,求出经过他的神犇路径的数量 题解: ...

  3. codeforces 1065F Up and Down the Tree

    题目链接:codeforces 1065F Up and Down the Tree 题意:给出一棵树的节点数\(n\)以及一次移动的最大距离\(k\),现在有一个标记在根节点1处,每一次可以进行一下 ...

  4. Codeforces 914H Ember and Storm's Tree Game 【DP】*

    Codeforces 914H Ember and Storm's Tree Game 题目链接 ORZ佬 果然出了一套自闭题 这题让你算出第一个人有必胜策略的方案数 然后我们就发现必胜的条件就是树上 ...

  5. Codeforces Round #499 (Div. 1) F. Tree

    Codeforces Round #499 (Div. 1) F. Tree 题目链接 \(\rm CodeForces\):https://codeforces.com/contest/1010/p ...

  6. CodeForces 1251B --- Binary Palindromes

    [CodeForces 1251B --- Binary Palindromes] Description A palindrome is a string t which reads the sam ...

  7. [Codeforces 553E]Kyoya and Train(期望DP+Floyd+分治FFT)

    [Codeforces 553E]Kyoya and Train(期望DP+Floyd+分治FFT) 题面 给出一个\(n\)个点\(m\)条边的有向图(可能有环),走每条边需要支付一个价格\(c_i ...

  8. codeforces 914E Palindromes in a Tree(点分治)

    You are given a tree (a connected acyclic undirected graph) of n vertices. Vertices are numbered fro ...

  9. Palindromes in a Tree CodeForces - 914E

    https://vjudge.net/problem/CodeForces-914E 点分就没一道不卡常的? 卡常记录: 1.把不知道为什么设的(unordered_map)s换成了(int[])s ...

随机推荐

  1. 软工网络15团队作业4-DAY8

    每日例会 昨天的工作. 张陈东芳:可导入部分类信息,继续尝试将所有信息导入: 吴敏烽:商品类的规范化编写: 周汉麟:界面的排版继续优化: 林振斌:按照浏览历史,次数等,继续优化商品类排序: 李智:研究 ...

  2. maven Web项目中POM的配置信息

    什么是POM? POM是项目对象模型(Project Object Model)的简称,它是Maven项目中的文件,使用XML表示,名称叫做pom.xml.在Maven中,当谈到Project的时候, ...

  3. session的基本原理

    转载:http://blog.sina.com.cn/s/blog_8155e74d0101iqmh.html 如何严格限制session在30分钟后过期! 1.设置客户端cookie的lifetim ...

  4. 第139天:详解cookie、 sessionStorage 和localStorage

    1.cookie:存储在用户本地终端上的数据.有时也用cookies,指某些网站为了辨别用户身份,进行session跟踪而存储在本地终端上的数据,通常经过加密.一般应用最典型的案列就是判断注册用户是否 ...

  5. h5端提示下载app

    // app下载提示 if (!sessionStorage.getItem("appDownloadTipClosed") && isAndroidOrIphon ...

  6. EM算法【转】

    混合高斯模型和EM算法 这篇讨论使用期望最大化算法(Expectation-Maximization)来进行密度估计(density estimation). 与K-means一样,给定的训练样本是, ...

  7. 【二】shiro入门 之 身份验证

    大体步骤如下: 1.首先通过new IniSecurityManagerFactory 并指定一个ini 配置文件来创建一个SecurityManager工厂: 2.接着获取SecurityManag ...

  8. BZOJ3591 最长上升子序列(状压dp)

    之前听说过一种dp套dp的trick,大致是用另一个dp过程中用到的一些东西作为该dp的状态.这个题比较类似. 考虑求LIS时用到的单调队列.设f[S]为所选取集合为S的方案数,其中在单调队列内的标2 ...

  9. Day 3 学习笔记

    Day 3 学习笔记 STL 模板库 一.结构体 结构体是把你所需要的一些自定义的类型(原类型.实例(:包括函数)的集合)都放到一个变量包里. 然后这个变量包与原先的类型差不多,可以开数组,是一种数据 ...

  10. Shortest Prefixes POJ - 2001(统计次数)

    题意: 输出每个单词的缩写  使得每个单词 互不相同.. 解析: 统计每个前出现的次数...然后在查询的时候  遇到次数为1的返回即可.. #include <iostream> #inc ...