模拟赛搬的题,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 可能有点常数。

代码

代码还是比较好写的。

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. typedef long long ll;
  4. typedef pair<int,int> pi;
  5. const ll eps=500005,mod=998244353;
  6. int n,k,a[15][50005],f[15][50005],tot[15][110005],r[15][50005],pr[50005],y[50005];
  7. ll hs[50005],dp[50005],ans;
  8. unordered_map<ll,int>mp;
  9. int main()
  10. {
  11. ios::sync_with_stdio(0);
  12. cin.tie(0);
  13. cout.tie(0);
  14. cin>>k>>n;
  15. //处理原括号序列、前缀和数组、各列的哈希值
  16. for(int i=1;i<=k;i++)
  17. {
  18. for(int j=1;j<=n;j++)
  19. {
  20. char c;
  21. cin>>c;
  22. if(c=='(')a[i][j]=1;
  23. else a[i][j]=-1;
  24. f[i][j]=f[i][j-1]+a[i][j];
  25. hs[j]=(hs[j]*10007%mod+f[i][j])%mod;
  26. }
  27. }
  28. //统计右边最远可达的括号
  29. memset(tot,0x3f,sizeof(tot));
  30. memset(r,0x3f,sizeof(r));
  31. for(int i=1;i<=k;i++)
  32. {
  33. for(int j=n;j>=1;j--)
  34. {
  35. tot[i][f[i][j]+eps]=j;
  36. r[i][j]=tot[i][f[i][j-1]-1+eps];
  37. }
  38. }
  39. //记录对于每一列而言的右边界
  40. memset(pr,0x3f,sizeof(pr));
  41. for(int i=1;i<=n;i++)
  42. {
  43. for(int j=1;j<=k;j++)
  44. {
  45. pr[i]=min(pr[i],r[j][i]);
  46. }
  47. }
  48. //找出相同的哈希值
  49. for(int i=n;i>=1;i--)
  50. {
  51. mp[hs[i]]=i;
  52. y[i]=mp[hs[i-1]];
  53. }
  54. //dp
  55. for(int i=n;i>=1;i--)
  56. {
  57. if(y[i]!=0&&y[i]<=pr[i])
  58. {
  59. dp[i]=dp[y[i]+1]+1;
  60. }
  61. }
  62. //统计答案
  63. for(int i=1;i<=n;i++)
  64. {
  65. ans+=dp[i];
  66. }
  67. cout<<ans;
  68. return 0;
  69. }

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. Graylog之基本使用

    文档:https://docs.graylog.org/en/3.0/ Graylog Sidecar是一个轻量级配置管理系统,适用于不同的日志收集器,也称为后端.Graylog节点充当包含日志收集器 ...

  2. spring ai 函数调用

    1.概要 我们使用AI大模型开发程序时,比如我需要查一下平台中有多少个客户.这个时候大模型肯定时不知道的,如果大模型不知道,他可能会回答不知道或者胡乱回答,这个时候就需要借助函数时调用来解决这些问题. ...

  3. 使用CANAL同步数据

    1.概要 canal 是阿里发布的一个mysql 同步工具,它是模拟 mysql slave 的方式读取binlog,并可以将数据写入到队列中. 如下图:是官方提供的架构图. 2.下载CANAL 下载 ...

  4. 借助AI助手快速解析LlamaIndex的Workflow设计与Java迁移

    在前面的讨论中,我们通过AI助手快速浏览并分析了LlamaIndex的核心源码及其可视化部分.在上次的工作中,我们已基本完成了使用Java版本实现的可视化部分,尽管在工作流(workflow)的分析上 ...

  5. 【位运算】codeforces 1775 C. Interesting Sequence

    题意 输入一个正整数 \(T(1 \leq T \leq 2000)\),代表 \(T\) 组测试用例.对于每个测试用例: 输入两个整数 \(n, m(0 \leq n, m \leq 10^{18} ...

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

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

  7. 中电资讯 - 一路“标”升,喜迎Q3开门红

    Q2收获满满,Q3精彩再启! 近日,中电金信多项业务取得新进展 接连中标多个项目 中电金信数字科技集团旗下优智汇咨询中标浙江民泰商业银行信息科技战略发展规划(2022-2025)项目.优智汇将根据民泰 ...

  8. 执行docker ps时提示"dial unix /var/run/docker.sock: connect: permission denied"

    0. 创建docker用户组 sudo groupadd docker 1. 将当前用户加入docker组 # sudo gpasswd -a $USER docker $ sudo usermod ...

  9. 如何设置AD域用户仅登录到指定的计算机?AD域管理软件

    一 什么是AD域? 简单理解:Active Directory域内的directory database(目录数据库)是被用来存储用户账户.计算机账户.打印机和共享文件夹等对象,而提供目录服务的组件就 ...

  10. FFmpeg中的色彩空间与像素格式3-像素格式

    FFmpeg 中的色彩与像素系列文章如下: [1]. FFmpeg中的色彩空间与像素格式1-色彩空间基础 [2]. FFmpeg中的色彩空间与像素格式2-RGB/YUV色彩空间 [3]. FFmpeg ...