洛谷 P5658 括号树 题解
简要题意:
求出以从每个节点到根形成的括号序列的合法对数。
算法一
观察到 \(n \leq 8\) ,所以我们可以用 纯粹的暴力 。
用 \(O(n)\) 时间得出当前节点到根的字符串。
然后 \(O(n^2)\) 枚举子串。
再用 \(O(n)\) 暴力判断(用栈)。
时间复杂度: \(O(n^5)\).
实际得分: \(10pts\).
优化一
用 \(s_i\) 表示 \(i\) 号节点对应的括号。
用 \(h_i\) 表示当前节点到根的字符串。
用 \(fa_i\) 表示当前节点的父亲编号。
则:
\]
由此,\(O(1)\) 推出字符串。
然后 \(O(n^2)\) 枚举子串,\(O(n)\) 验证。
时间复杂度: \(O(n^4)\)
实际得分: \(10pts\).
优化二
同样 \(O(1)\) 推出字符串。
下面,我们用 \(f_i\) 表示从第 \(i\) 号节点到根形成的括号序列的合法对数。
那么,显然存在:
\(f_i \geq f_{fa_i}\)
也就是说,我们只需要考虑以 \(i\) 号节点结尾的子串的贡献。
这样,我们枚举子串的时间复杂度降为 \(O(n)\),判断降为 \(O(n)\).
总时间复杂度: \(O(n^3)\)
实际得分:\(20pts\)
优化三
显然,我们需要摆脱枚举子串,而用另一些东西直接维护。
\begin{cases}
h_{fa_i},s_i = \text{(} \\
h_{fa_{g_i}}, s_i = \text{)} \\
\end{cases}
\]
其中, \(g_i\) 表示从当前节点开始,到根的路径上第一个(从下往上数)未匹配的左括号。
如果 \(s_i = \text{(}\) ,显然以当前节点结尾没有合法子串。
如果 \(s_i = \text{)}\) , 则找到它可以匹配的第一个左括号,然后记录那个左括号父亲的值(即前面的值)即可。
那么,\(g_i\) 如何维护呢?
\begin{cases}
i , s_i = \text{(} \\
g_{fa_i} , s_i = \text{)} \\
\end{cases}
\]
这也是显然的。
那么,我们可以做到 \(O(n)\) 时间维护这所有的东西。
总时间复杂度: \(O(n)\).
实际得分: \(100pts\).
#pragma GCC optimize(2)
#include<bits/stdc++.h>
using namespace std;
const int N=5e5+1;
typedef long long ll;
inline int read(){char ch=getchar();int f=1;while(ch<'0' || ch>'9') {if(ch=='-') f=-f; ch=getchar();}
int x=0;while(ch>='0' && ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();return x*f;}
char s[N];
int n; ll ans;
ll f[N],g[N],fa[N];
vector<int>G[N];
// f[i] 为 当前节点的贡献值
// g[i] 为 从当前节点起,第一个没匹配的左括号的编号
inline void dfs(int dep) {
g[dep]=g[fa[dep]];
if(s[dep]=='(') g[dep]=dep;
else if(g[dep]) f[dep]=f[fa[g[dep]]]+1,g[dep]=g[fa[g[dep]]]; //维护
for(int i=0;i<G[dep].size();i++) dfs(G[dep][i]); //递归下去
}
int main(){
n=read(); scanf("%s",s+1); //一个技巧,让字符串下标从 1 开始
for(int i=2,t;i<=n;i++) {
t=read(); fa[i]=t;
G[t].push_back(i); //存儿子节点编号
} dfs(1);
ans=f[1]; for(int i=2;i<=n;i++)
f[i]+=f[fa[i]],ans^=(i*f[i]); //我们把父亲的值放到最后算,避免出错
printf("%lld\n",ans);
return 0;
}
洛谷 P5658 括号树 题解的更多相关文章
- 洛谷 P5658 括号树
\(50pts\) #include <cstdio> #include <cstring> #include <iostream> #include <al ...
- P5658 括号树
P5658 括号树 题解 太菜了啥都不会写只能水5分数据 啥都不会写只能翻题解 题解大大我错了 我们手动找一下规律 我们设 w[ i ] 为从根节点到结点 i 对答案的贡献,也就是走到结点 i ,合 ...
- 2021.08.09 P5658 括号树(树形结构)
2021.08.09 P5658 括号树(树形结构) [P5658 CSP-S2019] 括号树 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 题意: 太长,在链接中. 分析及代码 ...
- 【洛谷P3960】列队题解
[洛谷P3960]列队题解 题目链接 题意: Sylvia 是一个热爱学习的女孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有 n×m ...
- 洛谷1087 FBI树 解题报告
洛谷1087 FBI树 本题地址:http://www.luogu.org/problem/show?pid=1087 题目描述 我们可以把由“0”和“1”组成的字符串分为三类:全“0”串称为B串,全 ...
- 洛谷P2832 行路难 分析+题解代码【玄学最短路】
洛谷P2832 行路难 分析+题解代码[玄学最短路] 题目背景: 小X来到了山区,领略山林之乐.在他乐以忘忧之时,他突然发现,开学迫在眉睫 题目描述: 山区有n座山.山之间有m条羊肠小道,每条连接两座 ...
- 洛谷P2312 解方程题解
洛谷P2312 解方程题解 题目描述 已知多项式方程: \[a_0+a_1x+a_2x^2+\cdots+a_nx^n=0\] 求这个方程在 \([1,m]\) 内的整数解(\(n\) 和 \(m\) ...
- 洛谷P1577 切绳子题解
洛谷P1577 切绳子题解 题目描述 有N条绳子,它们的长度分别为Li.如果从它们中切割出K条长度相同的 绳子,这K条绳子每条最长能有多长?答案保留到小数点后2位(直接舍掉2为后的小数). 输入输出格 ...
- 洛谷P2507 [SCOI2008]配对 题解(dp+贪心)
洛谷P2507 [SCOI2008]配对 题解(dp+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1299251 链接题目地址:洛谷P2507 [S ...
随机推荐
- 如何使用隐式转换扩展DataFrame和RDD以及其他的对象
目的 DataFrame可以点出来很多方法,都是DF内置的. 比如说:df.withColumn(),df.printSchema(). 但是如果你想打印df中的分区位置信息,以及每个key有多少记录 ...
- Flutter混合开发:Android接入Flutter
Flutter Google推出已经已经一年多了,单个 Flutter 项目的开发流程已经很成熟了.对与个人开发者来说使用 Flutter 开发一个跨平台的App挺有意思.但是对于现有的项目改造来说还 ...
- 用RecyclerView做一个小清新的Gallery效果 - Ryan Lee的博客
一.简介 RecyclerView现在已经是越来越强大,且不说已经被大家用到滚瓜烂熟的代替ListView的基础功能,现在RecyclerView还可以取代ViewPager实现Banner效果,当然 ...
- springboot利用swagger构建api文档
前言 Swagger 是一款RESTFUL接口的文档在线自动生成+功能测试功能软件.本文简单介绍了在项目中集成swagger的方法和一些常见问题.如果想深入分析项目源码,了解更多内容,见参考资料. S ...
- python 写入 execl 文件 之 xlwt 模块
1. xlwt 安装 pip install xlwt 2. xlwt 操作, 代码 #!/usr/bin/env python3 import xlwt # 只能创建 新的 excel 文件 # 1 ...
- PDF 相关操作
去年一年偷了下懒, 博客写了一点就没写了, 还好一些大的flag完成了. 花了半年的空余时间, 培养了一门兴趣爱好. 自己在为人处世上还是不够圆滑啊, 也难怪. 自己当初选择走技术这条路的初 ...
- Java中如何更优雅的处理空值
经常看到项目中存在到处空值判断的情况,这些判断,会让人觉得摸不着头绪,它的出现很有可能和当前的业务逻辑并没有关系.但它会让你很头疼.有时候,更可怕的是系统因为这些空值的情况,会抛出空指针异常,导致业务 ...
- sql05
1.Ado.net Ado.net是一组由微软提供的使用C#操作数据库的类库 2.连接 首先引入: using System.Data.SqlClient; 需要使用连接字符串进行连接 using S ...
- 编写简单i18n库
i18n是什么?i18n(其来源是英文单词internationalization的首末字符i和n,18为中间的字符数)是"国际化"的简称. 前言 第一次接触多语言是用野生java ...
- 关于毕业五年PHP成长疑惑
1.PHP语法基础是否都会,比如异常捕捉,面向对象,数组操作语法,字符串操作,cookie,session,全局变量,超全局数组,防止sql注入,mysql预处理 2.MYSQL基础语法,字段设计,原 ...