模拟赛搬的题,dp 思路很明显,但难点就在于找到要转移的点在哪。

暴力

首先我们可以先考虑 \(k=1\) 的情况,这应该很好想,就是对于每一个右括号,找到其匹配的左括号,然后进行转移即可,这个过程可以用栈维护。

\(dp[i]\) 定义为以 \(i\) 为结尾的合法序列个数。假设当前右括号在 \(i\) 处,匹配的左括号在 \(j\) 处,则:

\[dp[i]=dp[j-1]+1
\]

注意一定是要在保证能找到的情况下,转移离自己最近的左括号,才能保证所有括号序列都被统计到了。

最后扫一遍把所有的 \(dp[i]\) 累加即可。考场做法拿了 40pts。

正解

上面的做法,我们发现可以拓展到全局,也就是同时有 \(k\) 个序列的情况。

我们考虑一个括号序列的常用 trick:把左括号看作 \(+1\),把右括号看作 \(-1\),一个括号序列合法,当且仅当其总和为 \(0\) 且任何一段前缀和都 \(\ge 0\)。

总和为 \(0\) 很好考虑,我们主要想任何一段前缀和 \(\ge 0\) 怎么搞。

观察到前缀和数组每次相对前一项的变化量要么是 \(1\) 要么是 \(-1\),并且由于先要保证能找到,所以我们先找出可以匹配的左括号的区间左端点。

但是这样并不好做,因为如果 \([l,r]\) 的和 \(<0\),\([l-1,r]\) 的和却不一定 \(<0\)。所以固定右括号的方式不可行。

因此,我们才考虑固定左括号,去寻找右括号,并且把 dp 倒着做。

于是找出前缀和 \(<0\) 的就很简单了,对于一个左括号,其最多能匹配到的右括号一定在后面离自己最近的使前缀和 \(<0\) 的地方。

这个我们可以通过从后往前扫描,记录下考虑序列第 \(i\) 位到第 \(n\) 位里面前缀和为每一种数的最小下标,这样我们就可以快速查询在左括号后面,第一个使前缀和 \(<0\) 的右括号在哪了。

如果找不到,就说明右括号在右边的哪里都可以,所以赋为最大值。

算完最大右端点后,我们对于每一列,求出其最大右端点中的最小值,这就是某一列里可能的匹配范围。

接下来考虑总和为 \(0\) 的限制,很容易发现对于前缀和数组而言是这样的:

\[f[i]-f[j-1]=0
\]

可得:

\[f[i]=f[j-1]
\]

一个括号序列合法,必须每一行都满足这个条件,也就是说对于两个列而言,每一行的前缀和相同,它才可能合法。

所以我们对每一列哈希,存进 unordered_map,然后统计离自己最近的且在最大右端点左边的相同位即可。

最后来个 dp 就完事了,时间是 \(O(nk)\) 的,但 unordered_map 可能有点常数。

代码

代码还是比较好写的。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int> pi;
const ll eps=500005,mod=998244353;
int n,k,a[15][50005],f[15][50005],tot[15][110005],r[15][50005],pr[50005],y[50005];
ll hs[50005],dp[50005],ans;
unordered_map<ll,int>mp;
int main()
{
ios::sync_with_stdio(0);
cin.tie(0);
cout.tie(0);
cin>>k>>n;
//处理原括号序列、前缀和数组、各列的哈希值
for(int i=1;i<=k;i++)
{
for(int j=1;j<=n;j++)
{
char c;
cin>>c;
if(c=='(')a[i][j]=1;
else a[i][j]=-1;
f[i][j]=f[i][j-1]+a[i][j];
hs[j]=(hs[j]*10007%mod+f[i][j])%mod;
}
}
//统计右边最远可达的括号
memset(tot,0x3f,sizeof(tot));
memset(r,0x3f,sizeof(r));
for(int i=1;i<=k;i++)
{
for(int j=n;j>=1;j--)
{
tot[i][f[i][j]+eps]=j;
r[i][j]=tot[i][f[i][j-1]-1+eps];
}
}
//记录对于每一列而言的右边界
memset(pr,0x3f,sizeof(pr));
for(int i=1;i<=n;i++)
{
for(int j=1;j<=k;j++)
{
pr[i]=min(pr[i],r[j][i]);
}
}
//找出相同的哈希值
for(int i=n;i>=1;i--)
{
mp[hs[i]]=i;
y[i]=mp[hs[i-1]];
}
//dp
for(int i=n;i>=1;i--)
{
if(y[i]!=0&&y[i]<=pr[i])
{
dp[i]=dp[y[i]+1]+1;
}
}
//统计答案
for(int i=1;i<=n;i++)
{
ans+=dp[i];
}
cout<<ans;
return 0;
}

Luogu P3059 Concurrently Balanced Strings G 题解 [ 紫 ] [ 线性 dp ] [ 哈希 ] [ 括号序列 ]的更多相关文章

  1. [USACO12NOV]同时平衡线Concurrently Balanced Strings DP map 思维

    题面 [USACO12NOV]同时平衡线Concurrently Balanced Strings 题解 考虑DP. \(f[i]\)表示以\(i\)为左端点的合法区间个数.令\(pos[i]\)表示 ...

  2. [Usaco2012 Nov]Concurrently Balanced Strings

    Description [Brian Dean, 2012] Farmer John's cows are all of a very peculiar breed known for its dis ...

  3. 【区间DP】codevs3657 括号序列题解

    题目描述 Description 我们用以下规则定义一个合法的括号序列: (1)空序列是合法的 (2)假如S是一个合法的序列,则 (S) 和[S]都是合法的 (3)假如A 和 B 都是合法的,那么AB ...

  4. 【基础练习】【线性DP】codevs2622 数字序列(最大连续子序列和)题解

    版权信息 转载请注明出处 [ametake版权全部]http://blog.csdn.net/ametake欢迎来看 这道题目本质就是朴素的最大连续子序列和 直接上题目和代码 题目描写叙述 Descr ...

  5. 【AHOI2009】中国象棋 题解(线性DP+数学)

    前言:这题主要是要会设状态,状态找对了问题迎刃而解. --------------------------- 题目描述 这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可 ...

  6. 洛谷 P1360 [USACO07MAR]Gold Balanced Lineup G (前缀和+思维)

    P1360 [USACO07MAR]Gold Balanced Lineup G (前缀和+思维) 前言 题目链接 本题作为一道Stl练习题来说,还是非常不错的,解决的思维比较巧妙 算是一道不错的题 ...

  7. 线段树||BZOJ5194: [Usaco2018 Feb]Snow Boots||Luogu P4269 [USACO18FEB]Snow Boots G

    题面:P4269 [USACO18FEB]Snow Boots G 题解: 把所有砖和靴子排序,然后依次处理每一双靴子,把深度小于等于它的砖块都扔线段树里,问题就转化成了求线段树已有的砖块中最大的砖块 ...

  8. [LeetCode]1221. Split a String in Balanced Strings

    Balanced strings are those who have equal quantity of 'L' and 'R' characters. Given a balanced strin ...

  9. 【leetcode】1221. Split a String in Balanced Strings

    题目如下: Balanced strings are those who have equal quantity of 'L' and 'R' characters. Given a balanced ...

  10. 洛谷P3104 Counting Friends G 题解

    题目 [USACO14MAR]Counting Friends G 题解 这道题我们可以将 \((n+1)\) 个边依次去掉,然后分别判断去掉后是否能满足.注意到一点, \(n\) 个奶牛的朋友之和必 ...

随机推荐

  1. vant中dialog的使用

    按照文档上的方式引入组件,但是还是会报错说没有注册 引入方式如下: import { Cell, CellGroup, Field, Dialog } from 'vant'; components: ...

  2. HTML5 A链接

    1.基本使用 a标签常用属性: 属性名 说明 href 规定链接的目标 URL target 已什么形式打开这个连接 target属性有以下几个值 属性名 说明 _self 默认,当前页面跳转 _bl ...

  3. 【JS篇】控制子集超过一定数量开始轮播

    [JS篇]控制子集超过一定数量开始轮播, 这个是很早的时候的一个效果了,经过代码的不断迭代升级修改,现在是最封装的一版本,通过面向对象传参数,适用于任何一个需要放置 数量达到一定条件后可执行的函数 / ...

  4. Pro更改启动界面

    该方法适用于arcgispro 3.1及以上版本,我目前测试到3.3,是可以的. 使用的是pro产品的启动配置文件,利用其中的SplashScreen实现这一需求. 在bin目录下,新建(或编辑)Ar ...

  5. InstallShield软件详解

    InstallShield使用说明 文章目录 InstallShield使用说明 术语解释 工程介绍 InstallScript Basic MSI InstallScript MSI 如何选择适合的 ...

  6. 使用Tailwind CSS的几个小Tips

    前情 Tailwind CSS 是一个原子类 CSS 框架,它将基础的 CSS 全部拆分为原子级别.它的工作原理是扫描所有 HTML 文件.JavaScript 文件以及任何模板中的 CSS 类名,然 ...

  7. 一图归纳三大种类矩阵范数:诱导范数,元素范数,Schatten范数,涵盖谱范数,2范数

    转载自:https://blog.csdn.net/qq_27261889/article/details/87902480

  8. PM-企业数字化转型,数据治理是关键

      企业数字化转型是大势所趋,通过数据治理.数据平台建设.数据分析与建模,把数据变成服务,使数据能在企业内顺畅流动起来,为企业带来巨大的价值.数据是企业数字化转型的重要基础.   数据能创造价值,但数 ...

  9. 【C语言】【二级】移动一维数组中的内容;若数组中有n个整数,要求把下标从0到p的数组元素平移到数组的最后

    题目 请编写函数fun,函数的功能是:移动一维数组中的内容;若数组中有n个整数,要求把下标从0到p(含p, p小于等于n-1)的数组元素平移到数组的最后. 例如,一维数组中的原始内容为:1,2,3,4 ...

  10. 【爬虫】XPath实例

    题目要求我们用XPATH去爬某个网站并且保存为CSV文件 代码如下,仅供参考 # -*- coding: UTF-8 -*- # 开发人员:萌狼蓝天 # 博客:Https://mllt.cc # 笔记 ...