博客
关于我
【题解】括号树
阅读量: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/

你可能感兴趣的文章
MyEclipse设置当前行背景颜色、选中单词前景色、背景色
查看>>
myeclipse配置springmvc教程
查看>>
MyEclipse配置SVN
查看>>
MTCNN 人脸检测
查看>>
MyEcplise中SpringBoot怎样定制启动banner?
查看>>
MyPython
查看>>
MTD技术介绍
查看>>
MySQL
查看>>
MySQL
查看>>
mysql
查看>>
MTK Android 如何获取系统权限
查看>>
MySQL - 4种基本索引、聚簇索引和非聚索引、索引失效情况、SQL 优化
查看>>
MySQL - ERROR 1406
查看>>
mysql - 视图
查看>>
MySQL - 解读MySQL事务与锁机制
查看>>
MTTR、MTBF、MTTF的大白话理解
查看>>
mt_rand
查看>>
mysql /*! 50100 ... */ 条件编译
查看>>
mudbox卸载/完美解决安装失败/如何彻底卸载清除干净mudbox各种残留注册表和文件的方法...
查看>>
mysql 1264_关于mysql 出现 1264 Out of range value for column 错误的解决办法
查看>>