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

你可能感兴趣的文章
mysql主从复制及故障修复
查看>>
MySQL主从复制的原理和实践操作
查看>>
webpack loader配置全流程详解
查看>>
mysql主从复制,读写分离,半同步复制实现
查看>>
MySQL主从失败 错误Got fatal error 1236解决方法
查看>>
MySQL主从架构与读写分离实战
查看>>
MySQL主从篇:死磕主从复制中数据同步原理与优化
查看>>
mysql主从配置
查看>>
MySQL之2003-Can‘t connect to MySQL server on ‘localhost‘(10038)的解决办法
查看>>
MySQL之CRUD
查看>>
MySQL之DML
查看>>
Mysql之IN 和 Exists 用法
查看>>
MYSQL之REPLACE INTO和INSERT … ON DUPLICATE KEY UPDATE用法
查看>>
MySQL之SQL语句优化步骤
查看>>
MYSQL之union和order by分析([Err] 1221 - Incorrect usage of UNION and ORDER BY)
查看>>
Mysql之主从复制
查看>>
MySQL之函数
查看>>
mysql之分组查询GROUP BY,HAVING
查看>>
mysql之分页查询
查看>>
Mysql之备份与恢复
查看>>