博客
关于我
【题解】括号树
阅读量:187 次
发布时间:2019-02-28

本文共 1025 字,大约阅读时间需要 3 分钟。

在这里插入图片描述

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
#define int long longusing namespace std;const int N=5e5+5;char s[N];int n,fa[N],ans;int sta[N],Top,f[N],g[N];vector
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
0){ int t=sta[Top--]; g[x]=g[fa[t]]+1; f[x]=f[fa[x]]+g[x]; for(int i=0;i

转载地址:http://down.baihongyu.com/

你可能感兴趣的文章
MQTT 保留消息
查看>>
MQTT 持久会话与 Clean Session 详解
查看>>
MQTT工作笔记0007---剩余长度
查看>>
MQTT工作笔记0009---订阅主题和订阅确认
查看>>
Mqtt搭建代理服务器进行通信-浅析
查看>>
MS Edge浏览器“STATUS_INVALID_IMAGE_HASH“兼容性问题
查看>>
ms sql server 2008 sp2更新异常
查看>>
MS UC 2013-0-Prepare Tool
查看>>
MSBuild 教程(2)
查看>>
msbuild发布web应用程序
查看>>
MSB与LSB
查看>>
MSCRM调用外部JS文件
查看>>
MSCRM调用外部JS文件
查看>>
MSEdgeDriver (Chromium) 不适用于版本 >= 79.0.313 (Canary)
查看>>
MsEdgeTTS开源项目使用教程
查看>>
msf
查看>>
MSSQL数据库查询优化(一)
查看>>
MSSQL数据库迁移到Oracle(二)
查看>>
MSSQL日期格式转换函数(使用CONVERT)
查看>>
MSTP多生成树协议(第二课)
查看>>