【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)似乎要 ...
随机推荐
- Prometheus入门到放弃(7)之redis_exporter部署
redis监控,prometheus需要使用redis_exporter客户端. 这里我们采用docker方式部署,既可以部署在redis所在服务器,也可以部署在其他机器: docker镜像地址:ht ...
- makefile从0到1
一.什么是makefile 百度百科:Linux 环境下的程序员如果不会使用GNU make来构建和管理自己的工程,应该不能算是一个合格的专业程序员,至少不能称得上是Unix程序员.在 Linux(u ...
- 模块 logging random
模块logging logging模块的主要功能是记录软件调试.操作过程中的各种日志. 默认情况下Python的logging模块将日志打印到了标准输出中,且只显示了大于等于WARNING级别的日志, ...
- centos7编译安装memcached
1.libevent 源码地址:https://github.com/libevent/libevent/releases/download/release-2.1.8-stable/libevent ...
- Linux 环境安装 Node、nginx、docker、vsftpd、gitlab
Linux 环境安装 centos7 # 更新yum yum update -y 0. 防火墙 firewalld 新入的JD云服务器,发现防火墙默认是关闭的. # 查看防火墙状态 systemctl ...
- 关于PATCH与PUT的区别
两者的区别:PATCH:更新部分资源,非幂等,非安全PUT:更新整个资源,具有幂等性,非安全注:幂等性:多次请求的结果和请求一次的结果一样安全性:请求不改变资源状态 举个两者明显区别的例子(我对两者定 ...
- Java线程同步类容器和并发容器(四)
同步类容器都是线程安全的,在某些场景下,需要枷锁保护符合操作,最经典ConcurrentModifiicationException,原因是当容器迭代的过程中,被并发的修改了内容. for (Iter ...
- cocos版本说明
一直知道cocos是做游戏的,想学习一下,结果去官网一看就懵逼了.Cocos Creator,Cocos2d-x,cocos studio,Cocos2d-js,Cocos2d-x-lua,那一种才是 ...
- sql group by hour 按小时分组统计
Time字段以小时分组统计 select datepart(hour,time) hour,count(1) count from table where Similarity<75 group ...
- Java调用WebService方法总结(3)--wsimport调用WebService
wsimport是JDK自带的把WSDL转成Java的工具,可以很方便的生成调用WebService的代码.文中所使用到的软件版本:Java 1.8.0_191. 1.准备 参考Java调用WebSe ...