括号匹配(二)

时间限制:1000 ms  |  内存限制:65535 KB
难度:6
 
描述
   给你一个字符串,里面只包含"(",")","[","]"四种符号,请问你需要至少添加多少个括号才能使这些括号匹配起来。
如:
[]是匹配的
([])[]是匹配的
((]是不匹配的
([)]是不匹配的
 
输入
第一行输入一个正整数N,表示测试数据组数(N<=10)
每组测试数据都只有一行,是一个字符串S,S中只包含以上所说的四种字符,S的长度不超过100
输出
对于每组测试数据都输出一个正整数,表示最少需要添加的括号的数量。每组测试输出占一行
样例输入
4
[]
([])[]
((]
([)]
样例输出
0
0
3
2

对于这道题,我还是跟②2017=5.13-7.1中的合并石子一样,贴出两份不同的类型的题解以及解释。
思路其实差不多,是枚举过程有些差异,分给两种不同习惯的人。
DP[i][j]指i到j中合法括号配对数。比如对于字符串(),DP[1][2]=2。SP[]用于存储括号字符串,我是从下标1存储起的。N存储字符串的长度。
我们可以这样想,DP[i][j]合法的情况只有以下三种:
1.SP[i]=='('&&SP[j]==')'且DP[i+1][j-1]合法。
2.SP[i]=='['&&SP[j]==']'且DP[i+1][j-1]合法。3.能够切成两段合法的括号序列。 我们可以计算出合法的括号配对数,然后用字符串长度减去它,就是答案。这里合法的括号配对数指:比如对于字符串(),DP[1][2]=2,则合法括号配对数指2。
第一种题解:
 #include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
using namespace std;
int T,DP[][],N;
char SP[];
int main()
{
scanf("%d",&T);
while (T--) {
N=;
scanf("%s",SP+);
while (SP[N+]) N++;
memset(DP,,sizeof(DP));
for(int i=N-; i>=; i--)
for(int j=i+; j<=N; j++) {
if ((SP[i]=='('&&SP[j]==')')||(SP[i]=='['&&SP[j]==']'))
DP[i][j]=DP[i+][j-]+;
for (int k=i; k<j; k++)
DP[i][j]=max(DP[i][j],DP[i][k]+DP[k+][j]);
}
printf("%d\n",N-DP[][N]);
}
return ;
}

code1

我仍然挑出部分代码讲解:

memset(DP,0,sizeof(DP));//因为有T组数据,所以对于每一组数据我们都需要初始化DP数组。
for(int i=N-1; i>=1; i--)
  for(int j=i+1; j<=N; j++) {
    if ((SP[i]=='('&&SP[j]==')')||(SP[i]=='['&&SP[j]==']'))//第1、2种合法情况的判断。
      DP[i][j]=DP[i+1][j-1]+2;//括号有两边,所以+2;
    for (int k=i; k<j; k++)
      DP[i][j]=max(DP[i][j],DP[i][k]+DP[k+1][j]);//对于第三种情况的判断。
  }
printf("%d\n",N-DP[1][N]);

肯定有人要问:为什么DP[i][j]=max(DP[i][j],DP[i][k]+DP[k+1][j]);就可以计算出如([])()这样的情况的答案呢?

是这样的,我们可以这么想:假如k枚举到了3,那么DP[1][3]就是2,DP[4][6]也是2,DP[1][6]答案就是4。

假如k枚举到了4,那么DP[1][4]答案就是4,DP[5][6]答案就是2,DP[1][6]的答案就是6,大于原来的答案且合法,于是更新。

这样就可以完美地从括号中间分割出来啦~因为如果()()是类似这样的合法括号序列,从中间分割出来的答案一定是最优的,所以就可以保证答案的正确性啦!

这种写法的原理是:当i相同时,j小的先算;当i不相同时,i大的先算。

比如:DP[i][k]比DP[i][j]先算,因为i=i,k比较小;DP[k+1][j]比DP[i][j]先算,因为k+1不等于i,k+1>i。

这样就可以保证循环先后的正确性,也就是保证我们计算这一步需要用的东西在计算这一步之前就计算好了。

第二种写法:
 #include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
using namespace std;
int T,DP[][],N;
char SP[];
int main()
{
scanf("%d",&T);
while (T--) {
N=;
scanf("%s",SP+);
while (SP[N+]) N++;
memset(DP,,sizeof(DP));
for (int LEN=; LEN<=N; LEN++)
for (int i=,j=LEN; j<=N; i++,j++) {
if ((SP[i]=='(' && SP[j]==')')||(SP[i]=='['&&SP[j]==']'))
DP[i][j]=DP[i+][j-]+;
for (int k=i; k<j; k++)
DP[i][j]=max(DP[i][k]+DP[k+][j],DP[i][j]);
}
printf("%d\n",N-DP[][N]);
}
return ;
}

code2

这种写法其实本质上和第一种是一样的,只是枚举的方式略有不同,是通过枚举序列长度来保证DP循环的先后性的,所以放出来给熟悉这种写法的人看。


memset(DP,0,sizeof(DP));//初始化
for (int LEN=2; LEN<=N; LEN++) //枚举序列长度
  for (int i=1,j=LEN; j<=N; i++,j++) {
    if ((SP[i]=='(' && SP[j]==')')||(SP[i]=='['&&SP[j]==']'))//第1、2种合法情况的判断。
      DP[i][j]=DP[i+1][j-1]+2;//括号有两边,所以+2;
    for (int k=i; k<j; k++)
      DP[i][j]=max(DP[i][k]+DP[k+1][j],DP[i][j]);//对于第三种情况的判断。
  }
printf("%d\n",N-DP[1][N]);

对于第三种情况的判断上面讲的很清楚了。

有问题可以直接在评论里面提问,有需要转载的请得到我的允许,否则按侵权处理。


Elena loves NiroBC forever!

NYOJ15|括号匹配(二)|区间DP|Elena的更多相关文章

  1. POJ 2955 括号匹配,区间DP

    题意:给你一些括号,问匹配规则成立的括号的个数. 思路:这题lrj的黑书上有,不过他求的是添加最少的括号数,是的这些括号的匹配全部成立. 我想了下,其实这两个问题是一样的,我们可以先求出括号要匹配的最 ...

  2. poj2955:括号匹配,区间dp

    题目大意: 给一个由,(,),[,]组成的字符串,其中(),[]可以匹配,求最大匹配数 题解:区间dp: dp[i][j]表示区间 [i,j]中的最大匹配数 初始状态 dp[i][i+1]=(i,i+ ...

  3. POJ 2955 Brackets --最大括号匹配,区间DP经典题

    题意:给一段左右小.中括号串,求出这一串中最多有多少匹配的括号. 解法:此问题具有最优子结构,dp[i][j]表示i~j中最多匹配的括号,显然如果i,j是匹配的,那么dp[i][j] = dp[i+1 ...

  4. NYOJ15-括号匹配(二)-区间DP

    pid=15">http://acm.nyist.net/JudgeOnline/problem.php? pid=15 dp[i][j]表示从i到j至少须要加入多少个括号才干满足匹配 ...

  5. POJ - 2955 Brackets括号匹配(区间dp)

    Brackets We give the following inductive definition of a “regular brackets” sequence: the empty sequ ...

  6. NYOJ15括号匹配

    NYOJ15括号匹配 括号匹配(二) 时间限制:1000 ms  |  内存限制:65535 KB 难度:6   描述 给你一个字符串,里面只包含"(",")" ...

  7. 「kuangbin带你飞」专题二十二 区间DP

    layout: post title: 「kuangbin带你飞」专题二十二 区间DP author: "luowentaoaa" catalog: true tags: - ku ...

  8. 括号序列(区间dp)

    括号序列(区间dp) 输入一个长度不超过100的,由"(",")","[",")"组成的序列,请添加尽量少的括号,得到一 ...

  9. 【栈思想、DP】NYOJ-15 括号匹配(二)

    括号匹配(二) 描述 给你一个字符串,里面只包含"(",")","[","]"四种符号,请问你需要至少添加多少个括号才能 ...

随机推荐

  1. win8下C盘不能读写的解决方案[zz]

    做系统安全的时候发现了这个/setintegritylevel参数,没有找到更多资料,找到此文,看来这个参数有点神奇哟!我一个同事遇到了这个问题,主要症状:1.C 盘文件不能修改2.C 盘不能新建文件 ...

  2. 读取Excel二进制写入DB,并从DB中读取生成Excel文件

    namespace SendMailSMSService { class Program { static void Main(string[] args) { var connString = Sq ...

  3. 物联网架构成长之路(5)-EMQ插件配置

    1. 前言 上一小结说了插件的创建,这一节主要怎么编写代码,以及具体流程之类的.2. 增加一句Hello World 修改 ./deps/emq_plugin_wunaozai/src/emq_plu ...

  4. 【Linux高级驱动】input子系统框架

    [1.input子系统框架(drivers\input)] 如何得出某个驱动所遵循的框架?    1) 通过网络搜索    2) 自己想办法跟内核代码!         2.1 定位此驱动是属于哪种类 ...

  5. 【九天教您南方cass 9.1】 08 绘制等高线及对其处理

    同学们大家好,欢迎收看由老王测量上班记出品的cass9.1视频课程 我是本节课主讲老师九天. 我们讲课的教程附件也是共享的,请注意索取测量空间中. [点击索取cass教程]5元立得 (给客服说暗号:“ ...

  6. Java知多少(62)线程同步

    当两个或两个以上的线程需要共享资源,它们需要某种方法来确定资源在某一刻仅被一个线程占用.达到此目的的过程叫做同步(synchronization).像你所看到的,Java为此提供了独特的,语言水平上的 ...

  7. IntelliJ IDEA 中文乱码配置

    总共有下面几种乱码的解决方案: 工程乱码 执行main函数时,控制台乱码 运行tomcat时,控制台乱码 PS: 如果下面方案不生效时,打开IDEA安装目录找到 idea.exe.vmoptions( ...

  8. 在windows下安装git后没有ssh文件夹

    在windows7下安装git后,运行 cd ~/.ssh $ bash: cd: /c/Users/Administrator/.ssh: No such file or directory 出现以 ...

  9. Ubuntu 设置NAT共享网络(命令行方法)

    本文介绍如何使用iptables来实现NAT转发,事实上就是将一台机器作为网关(gateway)来使用.我们假设充当网关的机器至少有网卡eth0和eth1,使用eth0表示连接到外网的网卡,使用eth ...

  10. jQuery图片播放插件prettyPhoto使用介绍

    演示效果  http://www.17sucai.com/preview/131993/2014-07-09/mac-Bootstrap/gallery.html 点击之后的效果 使用方法 Query ...