【CSP-S 2019】【洛谷P5658】括号树【dfs】【二分】
题目:
题目链接:https://www.luogu.org/problem/P5658?contestId=24103
本题中合法括号串的定义如下:
()
是合法括号串。- 如果
A
是合法括号串,则(A)
是合法括号串。 - 如果
A
,B
是合法括号串,则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\) 的异或和,即:
\]
其中 \(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)\)
代码:
#include <stack>
#include <cstdio>
#include <vector>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
const int N=500010,Inf=1e9;
int n,tot,a[N],cnt[N][3],head[N];
ll ans[N],orz;
char ch;
vector<int> pos[N*2];
stack<int> del;
struct edge
{
int next,to;
}e[N];
void add(int from,int to)
{
e[++tot].to=to;
e[tot].next=head[from];
head[from]=tot;
}
int binary(int x,int tp)
{
int l=0,r=pos[x].size(),mid,res;
// for (int i=l;i<r;i++) printf("%d ",pos[x][i]);putchar(10);
while (l<=r)
{
mid=(l+r)>>1;
if (pos[x][mid]>=tp) r=mid-1,res=mid;
else l=mid+1;
}
return res;
}
void dfs(int x,int fa)
{
cnt[x][1]=cnt[fa][1]; cnt[x][2]=cnt[fa][2];
cnt[x][a[x]]++;
int s=cnt[x][1]-cnt[x][2]+N,pp=-1;
if (a[x]==1) del.push(x);
else
{
if (del.size()>1)
{
pp=del.top();
del.pop();
}
int tp=del.top();
pos[s].push_back(Inf);
ans[x]=pos[s].size()-binary(s,tp)-1;
pos[s].pop_back();
}
ans[x]+=ans[fa];
orz^=1LL*x*ans[x];
pos[s].push_back(x);
for (int i=head[x];~i;i=e[i].next)
dfs(e[i].to,x);
pos[s].pop_back();
if (del.top()==x) del.pop();
if (pp!=-1) del.push(pp);
}
int main()
{
memset(head,-1,sizeof(head));
scanf("%d",&n);
for (int i=1;i<=n;i++)
{
//while (ch=getchar()) if (ch=='('||ch==')') break;
while (1)
{
ch=getchar();
if (ch=='('||ch==')') break;
}
if (ch=='(') a[i]=1;
else a[i]=2;
}
for (int i=2,x;i<=n;i++)
{
scanf("%d",&x);
add(x,i);
}
del.push(-1);pos[N].push_back(0);
dfs(1,0);
printf("%lld\n",orz);
// for (int i=1;i<=n;i++)
// printf("%lld ",ans[i]);
return 0;
}
【CSP-S 2019】【洛谷P5658】括号树【dfs】【二分】的更多相关文章
- 洛谷 P5658 括号树 题解
原题链接 简要题意: 求出以从每个节点到根形成的括号序列的合法对数. 算法一 观察到 \(n \leq 8\) ,所以我们可以用 纯粹的暴力 . 用 \(O(n)\) 时间得出当前节点到根的字符串. ...
- 洛谷 P5658 括号树
\(50pts\) #include <cstdio> #include <cstring> #include <iostream> #include <al ...
- 括号树 noip(csp??) 2019 洛谷 P5658
洛谷AC通道 本题,题目长,但是实际想起来十分简单. 首先,对于树上的每一个后括号,我们很容易知道,他的贡献值等于上一个后括号的贡献值 + 1.(当然,前提是要有人跟他匹配,毕竟题目中要求了,是不同的 ...
- P5658 括号树
P5658 括号树 题解 太菜了啥都不会写只能水5分数据 啥都不会写只能翻题解 题解大大我错了 我们手动找一下规律 我们设 w[ i ] 为从根节点到结点 i 对答案的贡献,也就是走到结点 i ,合 ...
- 2021.08.09 P5658 括号树(树形结构)
2021.08.09 P5658 括号树(树形结构) [P5658 CSP-S2019] 括号树 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 题意: 太长,在链接中. 分析及代码 ...
- 洛谷1087 FBI树 解题报告
洛谷1087 FBI树 本题地址:http://www.luogu.org/problem/show?pid=1087 题目描述 我们可以把由“0”和“1”组成的字符串分为三类:全“0”串称为B串,全 ...
- 洛谷P3018 [USACO11MAR]树装饰Tree Decoration
洛谷P3018 [USACO11MAR]树装饰Tree Decoration树形DP 因为要求最小,我们就贪心地用每个子树中的最小cost来支付就行了 #include <bits/stdc++ ...
- 格雷码 CSP(NOIP??)2019 洛谷 P5657
洛谷AC通道! 多年过后,重新来看这道D1T1,20min不到AC,再回忆起当初考场三小时的抓耳挠腮,不禁感慨万千啊!! 发篇题解记录一下. 思路:直接dfs模拟即可(二进制找规律是不可能的, 这辈子 ...
- NOIP2017提高组Day2T3 列队 洛谷P3960 线段树
原文链接https://www.cnblogs.com/zhouzhendong/p/9265380.html 题目传送门 - 洛谷P3960 题目传送门 - LOJ#2319 题目传送门 - Vij ...
- 洛谷P3703 [SDOI2017]树点涂色(LCT,dfn序,线段树,倍增LCA)
洛谷题目传送门 闲话 这是所有LCT题目中的一个异类. 之所以认为是LCT题目,是因为本题思路的瓶颈就在于如何去维护同颜色的点的集合. 只不过做着做着,感觉后来的思路(dfn序,线段树,LCA)似乎要 ...
随机推荐
- Java中BIO,NIO,AIO的理解
在高性能的I/O体系设计中,有几个概念常常会使我们感到迷惑不解.具体如下: 1 什么是同步? 2 什么是异步? 3 什么是阻塞? 4 什么是非阻塞? 5 什么是同步阻塞? 6 什么是同步非阻塞? 7 ...
- 用Python写一个滑动验证码
1.准备阶段 滑动验证码我们可以直接用GEETEST的滑动验证码. 打开网址:https://www.geetest.com/ ,找到技术文档中的行为验证,打开部署文档,点击Python,下载ZIP包 ...
- Java 总结篇2
第02章:数据类型和运算符 一.概述: 1.数据类型:int.float.char.boolean 2.运算符:算术运算符.赋值运算符.关系运算符.逻辑运算符.位运算符(了解即可).条件运算符 3.基 ...
- MySQL练手小试题
创建表和数据 创建class表 create table class ( cid int(11) primary key auto_increment, caption varchar(32) not ...
- Android--TextView第一个单词大写
自定义TextView: public class FirstBoldTextView extends TextView { private boolean firstWordBold = false ...
- Fanuc Cnc 数控系统,PC端下发NC程序到CNC端,现场测试通过。
1.这几天把FANUC 数据采集(产量,状态,轴负载等),以及NC程序下发封装成独立的dll,方便其它项目调用,自己顺便写了下demo测试,在车间测试了几天,效果很好,完善了许多细节. 2.大概的界面 ...
- .Net MVC 输出HTML内容
1.后台代码中的带HTML标记的内容 ViewData["msg"]="<b>Title</b>"; 然则如许打印出来的就是 <b ...
- 【转载】 C#使用string.IsNullOrWhiteSpace方法判断字符串是否为非空字符
在C#编程过程中,很多时候需要判断传入过来的字符串是否为Null或者空字符或者空白字符,此时就可以使用到string.IsNullOrWhiteSpace方法来判断,如果字符串为null或者空字符Em ...
- 4.linux下配置Golang的环境变量
装好linux后优先在linux上配置Golang开发环境. 1.到Go语言中文网下载Linux安装包 https://studygolang.com/dl 2.到下载的目录下解压,下载的文件一般在“ ...
- 几种常见的Preference总结
DialogPreference共性 DialogPreference通用属性 说明 android:dialogIco 对话框的icon android:dialogLayout dialog的co ...