【题解】括号树

在这里插入图片描述

solution:

本题是个树上问题,但我们不妨从线性状态上去考虑。这题就是一道区间dp了。

问题:给定字符串S,求有多少合法括号子串?

()()(())

f [ i ] f[i] f[i]表示前 i i i个的合法子串个数, g [ i ] g[i] g[i]表示以 i i i结尾的合法字符串个数。

考虑如下转移:

g [ i ] = g [ k ] + 1 g[i]=g[k]+1 g[i]=g[k]+1

f [ i ] = f [ i − 1 ] + g [ i ] f[i]=f[i-1]+g[i] f[i]=f[i1]+g[i]

其中 [ k + 1 , i ] [k+1,i] [k+1,i]为合法子串。

例如,上图中, [ 5 , 8 ] [5,8] [5,8] ( ( ) ) (()) (())为合法子串,所以 g [ 8 ] = g [ 4 ] + 1 = 3 g[8]=g[4]+1=3 g[8]=g[4]+1=3 f [ 8 ] = f [ 7 ] + g [ 8 ] = 3 + 3 = 6 f[8]=f[7]+g[8]=3+3=6 f[8]=f[7]+g[8]=3+3=6

上述过程可以用栈模拟。时间复杂度是 O ( n ) O(n) O(n)

如何扩展到树上呢?我们知道, d f s dfs dfs是有回溯操作的,即一个分支结束后可以回到原始状态。这就很巧妙了,我们只需要用一个栈模拟回溯操作就行了。这样就得到了点到根的原括号序列 S S S O ( 1 ) O(1) O(1)转移,时间复杂度 O ( n ) O(n) O(n)

#include<bits/stdc++.h> #define int long long using namespace std; const int N=5e5+5; char s[N]; int n,fa[N],ans; int sta[N],Top,f[N],g[N]; vector<int> son[N]; void dfs(int x) { if(s[x]=='(') { sta[++Top]=x; f[x]=f[fa[x]]; g[x]=0; for(int i=0;i<son[x].size();i++) { int y=son[x][i]; dfs(y); } Top--; } else if(Top>0){ int t=sta[Top--]; g[x]=g[fa[t]]+1; f[x]=f[fa[x]]+g[x]; for(int i=0;i<son[x].size();i++) { int y=son[x][i]; dfs(y); } sta[++Top]=t; } else { f[x]=f[fa[x]]; g[x]=0; for(int i=0;i<son[x].size();i++) { int y=son[x][i]; dfs(y); } } ans^=(f[x]*x); } signed main() { scanf("%lld",&n); scanf("%s",s+1); for(int i=2;i<=n;i++) { scanf("%lld",&fa[i]); son[fa[i]].push_back(i); } dfs(1); printf("%lld",ans); }

__EOF__

本文作者仰望星空的蚂蚁
本文链接https://www.cnblogs.com/cqbzly/p/17530390.html
关于博主:评论和私信会在第一时间回复。或者直接私信我。
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。您的鼓励是博主的最大动力!
posted @   仰望星空的蚂蚁  阅读(11)  评论(0编辑  收藏  举报  
编辑推荐:
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· DeepSeek 开源周回顾「GitHub 热点速览」
点击右上角即可分享
微信分享提示