Brackets sequence UVA - 1626 (典型的区间dp+递归打印路径)

2024-04-29 13:32

本文主要是介绍Brackets sequence UVA - 1626 (典型的区间dp+递归打印路径),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

点击链接哦:https://vjudge.net/problem/51187/origin

题目大意:括弧的序列,在一个字符串中只包含" ( " " ) "和“ [ ” “ ] ”,要求空序列为正确的括弧,如果s是正确的序列,那么[s]和(s)也是正确的序列,如果a和b是是正确的序列,那么ab也是正确的序列。

求在给出的字符串基础上要至少添加多少括号才能使得字符序列是正确的。

ps:紫书p278

一开始自己先是用模拟去做的,自己感觉应该能可以模拟出来,而且长度为100也不会超时。但还是考虑的情况太少了,最终再无数的debug中把这个方法给pass掉,还是看了一下紫书的讲解。对于菜鸟的我在短时间内把这个题想到用dp,而且解决确实有点难度。看到转移方程时感觉就是通过题意把情况给遍历一遍一样。但是自己却没想到。

既然是区间dp,那么肯定是二维的了 设dp[i][j]表示至少需要增加的括号个数。

那么在区间i~j上,如果s[i]和s[j]匹配的话,那么最少增加的括号个数就是区间i+1~j-1上的了,那么如果不匹配的话,那就应该是有两个正确的序列组合而成,如果在区间i~j上有一个k,正确的序列是由dp[i][k]+dp[k+1][j]而来,(其实这只是构成正确序列的一种办法,还有其他的办法也能构成最少的正确的序列)。边界:当序列为空时dp[][]=0;当序列只有一个字符是,那么肯定要补上一个,显然dp[i][i]=1;然后用递推求出。在这里有一个关键,就是当s[i]和s[j]匹配的话,还需要通过第二种方案更新吗?如果不同动脑子的话我想为了保险需要比较一下看看能不能更优化。然而书上也是强调必须要在第二种方案上在比较更新一下。比如有序列" [ ] [ ] ",显然左右是匹配的,那么会得到序列“ ] [ ”,显然要多加两个括号,不合情理。

另一个重点就是打印,仍然是递归打印,具体看代码吧;

坑点:注意输入and输出

///典型的区间DP
#include <iostream>
#include <bits/stdc++.h>using namespace std;
string s;
int dp[111][111];
int match(char a,char b)
{if(a=='('&&b==')')return 1;if(a=='['&&b==']')return 1;return 0;
}
void print(int i,int j)
{if(i>j)return ;if(i==j){if(s[i]=='('||s[j]==')')printf("()");else printf("[]");return ;}int ans=dp[i][j];if(match(s[i],s[j])&&ans==dp[i+1][j-1]){printf("%c",s[i]);print(i+1,j-1);printf("%c",s[j]);return ;}for(int k=i;k<j;k++){if(ans==dp[i][k]+dp[k+1][j]){print(i,k);print(k+1,j);return ;}}
}
int main()
{int t;scanf("%d",&t);getchar();while(t--){getchar();getline(cin,s);int n=s.size();for(int i=0;i<n;i++){dp[i+1][i]=0;dp[i][i]=1;}for(int i=n-2;i>=0;i--){for(int j=i+1;j<n;j++){dp[i][j]=n;if(match(s[i],s[j]))dp[i][j]=min(dp[i][j],dp[i+1][j-1]);for(int k=i;k<j;k++){dp[i][j]=min(dp[i][j],dp[i][k]+dp[k+1][j]);}}}print(0,n-1);printf("\n");if(t)printf("\n");}return 0;
}


这篇关于Brackets sequence UVA - 1626 (典型的区间dp+递归打印路径)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/946224

相关文章

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使

SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志

《SpringBoot项目配置logback-spring.xml屏蔽特定路径的日志》在SpringBoot项目中,使用logback-spring.xml配置屏蔽特定路径的日志有两种常用方式,文中的... 目录方案一:基础配置(直接关闭目标路径日志)方案二:结合 Spring Profile 按环境屏蔽关

VSCode设置python SDK路径的实现步骤

《VSCode设置pythonSDK路径的实现步骤》本文主要介绍了VSCode设置pythonSDK路径的实现步骤,包括命令面板切换、settings.json配置、环境变量及虚拟环境处理,具有一定... 目录一、通过命令面板快速切换(推荐方法)二、通过 settings.json 配置(项目级/全局)三、

使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)

《使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)》字体设计和矢量图形处理是编程中一个有趣且实用的领域,通过Python的matplotlib库,我们可以轻松将字体轮廓... 目录背景知识字体轮廓的表示实现步骤1. 安装依赖库2. 准备数据3. 解析路径指令4. 绘制图形关键

如何更改pycharm缓存路径和虚拟内存分页文件位置(c盘爆红)

《如何更改pycharm缓存路径和虚拟内存分页文件位置(c盘爆红)》:本文主要介绍如何更改pycharm缓存路径和虚拟内存分页文件位置(c盘爆红)问题,具有很好的参考价值,希望对大家有所帮助,如有... 目录先在你打算存放的地方建四个文件夹更改这四个路径就可以修改默认虚拟内存分页js文件的位置接下来从高级-

Python多进程、多线程、协程典型示例解析(最新推荐)

《Python多进程、多线程、协程典型示例解析(最新推荐)》:本文主要介绍Python多进程、多线程、协程典型示例解析(最新推荐),本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定... 目录一、multiprocessing(多进程)1. 模块简介2. 案例详解:并行计算平方和3. 实现逻

一文详解如何查看本地MySQL的安装路径

《一文详解如何查看本地MySQL的安装路径》本地安装MySQL对于初学者或者开发人员来说是一项基础技能,但在安装过程中可能会遇到各种问题,:本文主要介绍如何查看本地MySQL安装路径的相关资料,需... 目录1. 如何查看本地mysql的安装路径1.1. 方法1:通过查询本地服务1.2. 方法2:通过MyS

PostgreSQL 序列(Sequence) 与 Oracle 序列对比差异分析

《PostgreSQL序列(Sequence)与Oracle序列对比差异分析》PostgreSQL和Oracle都提供了序列(Sequence)功能,但在实现细节和使用方式上存在一些重要差异,... 目录PostgreSQL 序列(Sequence) 与 oracle 序列对比一 基本语法对比1.1 创建序

mysql递归查询语法WITH RECURSIVE的使用

《mysql递归查询语法WITHRECURSIVE的使用》本文主要介绍了mysql递归查询语法WITHRECURSIVE的使用,WITHRECURSIVE用于执行递归查询,特别适合处理层级结构或递归... 目录基本语法结构:关键部分解析:递归查询的工作流程:示例:员工与经理的层级关系解释:示例:树形结构的数

Python如何调用指定路径的模块

《Python如何调用指定路径的模块》要在Python中调用指定路径的模块,可以使用sys.path.append,importlib.util.spec_from_file_location和exe... 目录一、sys.path.append() 方法1. 方法简介2. 使用示例3. 注意事项二、imp