【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)似乎要 ...
随机推荐
- 计算GPA
#include <stdio.h> int main() { int n,t,i; float sum,s,p,m,k; while(~scanf("%d",& ...
- Testbench编写技巧
一.基本架构(常用模板) `timescale 1ns/1ps //时间精度 `define Clock //时钟周期 module my_design_tb; //================= ...
- IntelliJ IDEA 2019 激活码 | 全产品 | 跨平台 | Goland | PhpStorm | Rider | CentOS | Windows
>>> 下载地址: https://kenkao.pipipan.com/fs/14896800-375468824 >>> 下载地址2: https://pan. ...
- Oracle 11g安装过程工作Oracle数据库安装图解
一.Oracle 下载 注意Oracle分成两个文件,下载完后,将两个文件解压到同一目录下即可. 路径名称中,最好不要出现中文,也不要出现空格等不规则字符. 官方下地址: oracle.com/tec ...
- chocolatey install curl and netcat
chocolatey install curl and netcat 软件仓库 https://chocolatey.org/packages choco install curl choco ins ...
- 使用springboot实现一个简单的restful crud——02、dao层单元测试,测试从数据库取数据
接着上一篇,上一篇我们创建了项目.创建了实体类,以及创建了数据库数据.这一篇就写一下Dao层,以及对Dao层进行单元测试,看下能否成功操作数据库数据. Dao EmpDao package com.j ...
- 【洛谷 P4302】 [SCOI2003]字符串折叠(DP)
题目链接 简单区间dp 令\(f[i][j]\)表示\([i,j]\)的最短长度 先枚举区间,然后在区间中枚举长度\(k\),看这个区间能不能折叠成几个长度为\(k\)的,如果能就更新答案. #inc ...
- 【转载】 C#中List集合使用First()方法获取第一个元素
在C#的List集合操作过程中,如果要获取List集合中的第一个元素对象,则一般会先通过获取到list[0]这种方式来获取第一个元素.其实在List集合中提供了获取最后一个元素的First()方法,调 ...
- AndroidStudio中Run按钮是灰色的解决方法
在model下拉框中选择app.如果下拉框中没有app,(没有工程名),那么请先去设置: Android Studio 3.3.0 File->sync project with gradles ...
- 尚硅谷韩顺平Linux教程学习笔记
目录 尚硅谷韩顺平Linux教程学习笔记 写在前面 虚拟机 Linux目录结构 远程登录Linux系统 vi和vim编辑器 关机.重启和用户登录注销 用户管理 实用指令 组管理和权限管理 定时任务调度 ...