POJ1141 Brackets Sequence (dp动态规划,递归)

2024-02-29 03:48

本文主要是介绍POJ1141 Brackets Sequence (dp动态规划,递归),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文出自:http://blog.csdn.net/svitter

原题:http://poj.org/problem?id=1141

题意:输出添加括号最少,并且使其匹配的串。


题解: dp [ i ] [ j ] 表示添加括号的个数, pos[ i][ j ] 表示 i , j 中哪个位置分开,使得两部分分别匹配。

pos [ i ][ j ] 为-1的时候,说明i, j 括号匹配。

初始值置dp [ i ] [ i ]  = 1; 如果只有一个括号,那么匹配结果必然是差1。

首先判断括号是否匹配,如果匹配,那么dp [ i ] [ j ] = dp[ i + 1] [ j - 1] 。如此递推dp [ i ] [ j ] 的值。

然后判断dp [ i ] [ j ] = min ( dp [ i ] [ mid ] + dp [ mid + 1 ] [ j ]);

for k = 1...len

 for i = 0...i + k < len

  求出dp[ i ] [ i + k ]每段。先求短区间,再求长区间,用短区间来求长区间。

可以说是http://blog.csdn.net/svitter/article/details/24877159的加强变种。

//注:除了多添加一个pos其他的一模一样。但是用当初的算法显然不符合此时的条件,只得作罢。后来借鉴了别人的代码,才推出。

再一个问题就是print_str(i , j)函数。按区间输出。、

//注:开始想到pos时,觉得没法输出放弃了,后来看到这个算法才算明白。

依据分段来输出,pos如果不为-1,那么需要从pos处分开输出。

如果pos为-1,那么输出首括号,输出中间部分,输出尾括号(两者是匹配的)

如果i == j , 如果括号为( | )则输出(), 如果括号为[ | ]则输出 [] 。

注意:必须用gets函数,因为输入数据中有空格,使用scanf函数中间会出问题。

AC代码:

//============================================================================
// Name        : test.cpp
// Author      : 
// Version     :
// Copyright   : Your copyright notice
// Description : Hello World in C++, Ansi-style
//============================================================================
//============================================================================
// Name        : 动态规划.cpp
// Author      : blog.csdn.net/svitter
// Version     :
// Copyright   : Your copyright notice
// Description : Hello World in C++, Ansi-style
//============================================================================#include <iostream>
#include <stdio.h>
#include <string.h>using namespace std;
#define MAXN 256
char br[MAXN];
int dp[MAXN][MAXN], pos[MAXN][MAXN];
int len;void print_br(int i, int j){if(i > j)return;if(i == j){if(br[i] =='(' || br[j] == ')')printf("()");elseprintf("[]");}else if(pos[i][j] == -1){printf("%c", br[i]);print_br(i+1, j-1);printf("%c", br[j]);}else{print_br(i, pos[i][j]);print_br(pos[i][j] + 1, j);}
}bool match(int i, int j)
{if(br[i] == '(' && br[j] == ')')return true;if(br[i] == '[' && br[j] == ']')return true;return false;
}int main() {//work pitint i, j, k, mid, t;while (gets(br) != NULL) {memset(dp, 0, sizeof(dp));len = strlen(br);for (i = 0; i < len; i++)dp[i][i] = 1;for (k = 1; k < len; k++) {for(i = 0; i + k < len; i ++){j = i + k;dp[i][j] = 0x7fffffff;if(match(i, j)){//如果当前位置匹配,那么pos置-1dp[i][j] = dp[i+1][j-1] , pos[i][j] = -1;}for(mid = i;  mid < j; mid++){if(dp[i][j] > (t = dp[i][mid] + dp[mid+1][j])){//如果存在更优分解,那么选择更优分解dp[i][j] = t, pos[i][j] = mid;}}}}print_br(0, len - 1);printf("\n");}return 0;
}


这篇关于POJ1141 Brackets Sequence (dp动态规划,递归)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

hdu4826(三维DP)

这是一个百度之星的资格赛第四题 题目链接:http://acm.hdu.edu.cn/contests/contest_showproblem.php?pid=1004&cid=500 题意:从左上角的点到右上角的点,每个点只能走一遍,走的方向有三个:向上,向下,向右,求最大值。 咋一看像搜索题,先暴搜,TLE,然后剪枝,还是TLE.然后我就改方法,用DP来做,这题和普通dp相比,多个个向上

hdu1011(背包树形DP)

没有完全理解这题, m个人,攻打一个map,map的入口是1,在攻打某个结点之前要先攻打其他一个结点 dp[i][j]表示m个人攻打以第i个结点为根节点的子树得到的最优解 状态转移dp[i][ j ] = max(dp[i][j], dp[i][k]+dp[t][j-k]),其中t是i结点的子节点 代码如下: #include<iostream>#include<algorithm

动态规划---打家劫舍

题目: 你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。 给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。 思路: 动态规划五部曲: 1.确定dp数组及含义 dp数组是一维数组,dp[i]代表

hdu4865(概率DP)

题意:已知前一天和今天的天气概率,某天的天气概率和叶子的潮湿程度的概率,n天叶子的湿度,求n天最有可能的天气情况。 思路:概率DP,dp[i][j]表示第i天天气为j的概率,状态转移如下:dp[i][j] = max(dp[i][j, dp[i-1][k]*table2[k][j]*table1[j][col] )  代码如下: #include <stdio.h>#include

usaco 1.1 Broken Necklace(DP)

直接上代码 接触的第一道dp ps.大概的思路就是 先从左往右用一个数组在每个点记下蓝或黑的个数 再从右到左算一遍 最后取出最大的即可 核心语句在于: 如果 str[i] = 'r'  ,   rl[i]=rl[i-1]+1, bl[i]=0 如果 str[i] = 'b' ,  bl[i]=bl[i-1]+1, rl[i]=0 如果 str[i] = 'w',  bl[i]=b

软考系统规划与管理师考试证书含金量高吗?

2024年软考系统规划与管理师考试报名时间节点: 报名时间:2024年上半年软考将于3月中旬陆续开始报名 考试时间:上半年5月25日到28日,下半年11月9日到12日 分数线:所有科目成绩均须达到45分以上(包括45分)方可通过考试 成绩查询:可在“中国计算机技术职业资格网”上查询软考成绩 出成绩时间:预计在11月左右 证书领取时间:一般在考试成绩公布后3~4个月,各地领取时间有所不同

uva 10154 DP 叠乌龟

题意: 给你几只乌龟,每只乌龟有自身的重量和力量。 每只乌龟的力量可以承受自身体重和在其上的几只乌龟的体重和内。 问最多能叠放几只乌龟。 解析: 先将乌龟按力量从小到大排列。 然后dp的时候从前往后叠,状态转移方程: dp[i][j] = dp[i - 1][j];if (dp[i - 1][j - 1] != inf && dp[i - 1][j - 1] <= t[i]

uva 10118 dP

题意: 给4列篮子,每次从某一列开始无放回拿蜡烛放入篮子里,并且篮子最多只能放5支蜡烛,数字代表蜡烛的颜色。 当拿出当前颜色的蜡烛在篮子里存在时,猪脚可以把蜡烛带回家。 问最多拿多少只蜡烛。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cs

uva 10069 DP + 大数加法

代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vector>#include <queue>#include <map>#include <cl