题目:

题目链接:https://www.luogu.org/problem/P5658?contestId=24103

本题中合法括号串的定义如下:

  1. () 是合法括号串。
  2. 如果 A 是合法括号串,则 (A) 是合法括号串。
  3. 如果 AB 是合法括号串,则 AB 是合法括号串。

本题中子串不同的子串的定义如下:

4. 字符串 S 的子串是 S连续的任意个字符组成的字符串。S 的子串可用起始位置 \(l\) 与终止位置 \(r\) 来表示,记为 \(S (l, r)\)(\(1 \leq l \leq r \leq |S |\),\(|S |\) 表示 S 的长度)。

5. S 的两个子串视作不同当且仅当它们在 S 中的位置不同,即 \(l\) 不同或 \(r\) 不同。

一个大小为 \(n\) 的树包含 \(n\) 个结点和 \(n − 1\) 条边,每条边连接两个结点,且任意两个结点间有且仅有一条简单路径互相可达。

小 Q 是一个充满好奇心的小朋友,有一天他在上学的路上碰见了一个大小为 \(n\) 的树,树上结点从 \(1\) ∼ \(n\) 编号,\(1\) 号结点为树的根。除 \(1\) 号结点外,每个结点有一个父亲结点,\(u\)(\(2 \leq u \leq n\))号结点的父亲为 \(f_u\)(\(1 ≤ f_u < u\))号结点。

小 Q 发现这个树的每个结点上恰有一个括号,可能是()。小 Q 定义 \(s_i\) 为:将根结点到 \(i\) 号结点的简单路径上的括号,按结点经过顺序依次排列组成的字符串。

显然 \(s_i\) 是个括号串,但不一定是合法括号串,因此现在小 Q 想对所有的 \(i\)(\(1\leq i\leq n\))求出,\(s_i\) 中有多少个互不相同的子串合法括号串

这个问题难倒了小 Q,他只好向你求助。设 \(s_i\) 共有 \(k_i\) 个不同子串是合法括号串, 你只需要告诉小 Q 所有 \(i \times k_i\) 的异或和,即:

\[(1 \times k_1)\ \text{xor}\ (2 \times k_2)\ \text{xor}\ (3 \times k_3)\ \text{xor}\ \cdots\ \text{xor}\ (n \times k_n)
\]

其中 \(xor\) 是位异或运算。

思路:

我们设\(ans[x]\)表示路径\((1,x)\)中构成的括号串,以\(x\)节点为右端点的所有区间有多少个合法括号串。

  • 那么如果\(x\)位置为(,那么显然\(ans[x]=0\)。
  • 如果\(x\)位置为),设\(cnt[x][1/2]\)为路径\((1,x)\)中左括号和右括号的个数,那么一个\(x\)节点的祖先\(y\)可以对\(ans[x]\)做贡献,当且仅当满足一下两个条件:

    \((1)\ cnt[x][1]-cnt[y][1]=cnt[x][2]-cnt[y][2]\)

    \((2)\ ∀p\in(y,x)\),满足\(cnt[p][2]\geq cnt[p][1]\)

那么我们就可以在访问每一个节点时,依次枚举它的每一个祖先,如果满足\(cnt[x][1]-cnt[y][1]=cnt[x][2]-cnt[y][2]\),那么\(ans[x]++\)。直到\(cnt[y][2]< cnt[y][1]\)时停止枚举。

这样我们就得到了一个\(O(n^2)\)的算法,获得了\(50pts\)的好成绩。

我们发现,其实我们只关心在路径\((1,x)\)中,深度最大的不满足\(cnt[p][2]\geq cnt[p][1]\)的节点\(p\)是哪一个。这样所有在路径\((son[p],x)\)中满足条件\((1)\)的点都可以做贡献。

其实\((1)\)的条件可以转化为\(cnt[x][1]-cnt[x][2]=cnt[y][1]-cnt[y][2]\)。所以我们可以用\(pos[s][tot]\)记录\(cnt[y][1]-cnt[y][2]=s\)的每一个\(x\)的祖先\(y\)编号。这样如果\(cnt[x][1]-cnt[x][2]=s\),那么能对\(x\)做贡献的点就都在\(pos[s]\)中。

那么我们可以用一个栈来记录\(cnt[p][2]<cnt[p][1]\)的所有\(p\)。其中\(p\)是\(x\)的祖先。此时如果节点\(x\)为(,那么直接将\(x\)扔进栈里。如果\(x\)为),那么就弹出栈顶。

这样如果栈顶是\(p\),那么能对\(x\)做贡献的就是同时在路径\((son[p],x)\)和\(pos[cnt[x][1]-cnt[x][2]\)的节点。

所以就可以二分出\(ans[x]\)。

发现\(pos\)中最多只会有\(n\)个元素,所以可以开一个\(vector\)。

求出\(ans[x]\)后,路径\((1,x)\)的合法括号串个数就是\(\sum^{y\texttt{是}x\texttt{的祖先}}_{y}ans[y]\)。做前缀和即可。

注意回溯时需要在栈中弹出\(x\)。

时间复杂度\(O(n\log n)\)

代码:

  1. #include <stack>
  2. #include <cstdio>
  3. #include <vector>
  4. #include <cstring>
  5. #include <algorithm>
  6. using namespace std;
  7. typedef long long ll;
  8. const int N=500010,Inf=1e9;
  9. int n,tot,a[N],cnt[N][3],head[N];
  10. ll ans[N],orz;
  11. char ch;
  12. vector<int> pos[N*2];
  13. stack<int> del;
  14. struct edge
  15. {
  16. int next,to;
  17. }e[N];
  18. void add(int from,int to)
  19. {
  20. e[++tot].to=to;
  21. e[tot].next=head[from];
  22. head[from]=tot;
  23. }
  24. int binary(int x,int tp)
  25. {
  26. int l=0,r=pos[x].size(),mid,res;
  27. // for (int i=l;i<r;i++) printf("%d ",pos[x][i]);putchar(10);
  28. while (l<=r)
  29. {
  30. mid=(l+r)>>1;
  31. if (pos[x][mid]>=tp) r=mid-1,res=mid;
  32. else l=mid+1;
  33. }
  34. return res;
  35. }
  36. void dfs(int x,int fa)
  37. {
  38. cnt[x][1]=cnt[fa][1]; cnt[x][2]=cnt[fa][2];
  39. cnt[x][a[x]]++;
  40. int s=cnt[x][1]-cnt[x][2]+N,pp=-1;
  41. if (a[x]==1) del.push(x);
  42. else
  43. {
  44. if (del.size()>1)
  45. {
  46. pp=del.top();
  47. del.pop();
  48. }
  49. int tp=del.top();
  50. pos[s].push_back(Inf);
  51. ans[x]=pos[s].size()-binary(s,tp)-1;
  52. pos[s].pop_back();
  53. }
  54. ans[x]+=ans[fa];
  55. orz^=1LL*x*ans[x];
  56. pos[s].push_back(x);
  57. for (int i=head[x];~i;i=e[i].next)
  58. dfs(e[i].to,x);
  59. pos[s].pop_back();
  60. if (del.top()==x) del.pop();
  61. if (pp!=-1) del.push(pp);
  62. }
  63. int main()
  64. {
  65. memset(head,-1,sizeof(head));
  66. scanf("%d",&n);
  67. for (int i=1;i<=n;i++)
  68. {
  69. //while (ch=getchar()) if (ch=='('||ch==')') break;
  70. while (1)
  71. {
  72. ch=getchar();
  73. if (ch=='('||ch==')') break;
  74. }
  75. if (ch=='(') a[i]=1;
  76. else a[i]=2;
  77. }
  78. for (int i=2,x;i<=n;i++)
  79. {
  80. scanf("%d",&x);
  81. add(x,i);
  82. }
  83. del.push(-1);pos[N].push_back(0);
  84. dfs(1,0);
  85. printf("%lld\n",orz);
  86. // for (int i=1;i<=n;i++)
  87. // printf("%lld ",ans[i]);
  88. return 0;
  89. }

【CSP-S 2019】【洛谷P5658】括号树【dfs】【二分】的更多相关文章

  1. 洛谷 P5658 括号树 题解

    原题链接 简要题意: 求出以从每个节点到根形成的括号序列的合法对数. 算法一 观察到 \(n \leq 8\) ,所以我们可以用 纯粹的暴力 . 用 \(O(n)\) 时间得出当前节点到根的字符串. ...

  2. 洛谷 P5658 括号树

    \(50pts\) #include <cstdio> #include <cstring> #include <iostream> #include <al ...

  3. 括号树 noip(csp??) 2019 洛谷 P5658

    洛谷AC通道 本题,题目长,但是实际想起来十分简单. 首先,对于树上的每一个后括号,我们很容易知道,他的贡献值等于上一个后括号的贡献值 + 1.(当然,前提是要有人跟他匹配,毕竟题目中要求了,是不同的 ...

  4. P5658 括号树

    P5658 括号树 题解 太菜了啥都不会写只能水5分数据 啥都不会写只能翻题解  题解大大我错了 我们手动找一下规律 我们设 w[ i ] 为从根节点到结点 i 对答案的贡献,也就是走到结点 i ,合 ...

  5. 2021.08.09 P5658 括号树(树形结构)

    2021.08.09 P5658 括号树(树形结构) [P5658 CSP-S2019] 括号树 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 题意: 太长,在链接中. 分析及代码 ...

  6. 洛谷1087 FBI树 解题报告

    洛谷1087 FBI树 本题地址:http://www.luogu.org/problem/show?pid=1087 题目描述 我们可以把由“0”和“1”组成的字符串分为三类:全“0”串称为B串,全 ...

  7. 洛谷P3018 [USACO11MAR]树装饰Tree Decoration

    洛谷P3018 [USACO11MAR]树装饰Tree Decoration树形DP 因为要求最小,我们就贪心地用每个子树中的最小cost来支付就行了 #include <bits/stdc++ ...

  8. 格雷码 CSP(NOIP??)2019 洛谷 P5657

    洛谷AC通道! 多年过后,重新来看这道D1T1,20min不到AC,再回忆起当初考场三小时的抓耳挠腮,不禁感慨万千啊!! 发篇题解记录一下. 思路:直接dfs模拟即可(二进制找规律是不可能的, 这辈子 ...

  9. NOIP2017提高组Day2T3 列队 洛谷P3960 线段树

    原文链接https://www.cnblogs.com/zhouzhendong/p/9265380.html 题目传送门 - 洛谷P3960 题目传送门 - LOJ#2319 题目传送门 - Vij ...

  10. 洛谷P3703 [SDOI2017]树点涂色(LCT,dfn序,线段树,倍增LCA)

    洛谷题目传送门 闲话 这是所有LCT题目中的一个异类. 之所以认为是LCT题目,是因为本题思路的瓶颈就在于如何去维护同颜色的点的集合. 只不过做着做着,感觉后来的思路(dfn序,线段树,LCA)似乎要 ...

随机推荐

  1. 计算GPA

    #include <stdio.h> int main() { int n,t,i; float sum,s,p,m,k; while(~scanf("%d",& ...

  2. Testbench编写技巧

    一.基本架构(常用模板) `timescale 1ns/1ps //时间精度 `define Clock //时钟周期 module my_design_tb; //================= ...

  3. IntelliJ IDEA 2019 激活码 | 全产品 | 跨平台 | Goland | PhpStorm | Rider | CentOS | Windows

    >>> 下载地址: https://kenkao.pipipan.com/fs/14896800-375468824 >>> 下载地址2: https://pan. ...

  4. Oracle 11g安装过程工作Oracle数据库安装图解

    一.Oracle 下载 注意Oracle分成两个文件,下载完后,将两个文件解压到同一目录下即可. 路径名称中,最好不要出现中文,也不要出现空格等不规则字符. 官方下地址: oracle.com/tec ...

  5. chocolatey install curl and netcat

    chocolatey install curl and netcat 软件仓库 https://chocolatey.org/packages choco install curl choco ins ...

  6. 使用springboot实现一个简单的restful crud——02、dao层单元测试,测试从数据库取数据

    接着上一篇,上一篇我们创建了项目.创建了实体类,以及创建了数据库数据.这一篇就写一下Dao层,以及对Dao层进行单元测试,看下能否成功操作数据库数据. Dao EmpDao package com.j ...

  7. 【洛谷 P4302】 [SCOI2003]字符串折叠(DP)

    题目链接 简单区间dp 令\(f[i][j]\)表示\([i,j]\)的最短长度 先枚举区间,然后在区间中枚举长度\(k\),看这个区间能不能折叠成几个长度为\(k\)的,如果能就更新答案. #inc ...

  8. 【转载】 C#中List集合使用First()方法获取第一个元素

    在C#的List集合操作过程中,如果要获取List集合中的第一个元素对象,则一般会先通过获取到list[0]这种方式来获取第一个元素.其实在List集合中提供了获取最后一个元素的First()方法,调 ...

  9. AndroidStudio中Run按钮是灰色的解决方法

    在model下拉框中选择app.如果下拉框中没有app,(没有工程名),那么请先去设置: Android Studio 3.3.0 File->sync project with gradles ...

  10. 尚硅谷韩顺平Linux教程学习笔记

    目录 尚硅谷韩顺平Linux教程学习笔记 写在前面 虚拟机 Linux目录结构 远程登录Linux系统 vi和vim编辑器 关机.重启和用户登录注销 用户管理 实用指令 组管理和权限管理 定时任务调度 ...