标签:
Time Limit: 1000MS | Memory Limit: 65536K | |||
Total Submissions: 29236 | Accepted: 8319 | Special Judge |
Description
Input
Output
Sample Input
([(]
Sample Output
()[()]
Source
#include<string.h> #include<stdio.h> #include <iostream> using namespace std; #define N 50 int dp[N][N]; ///dp[i][j] 代表 AiAi+1...Aj相乘的最小代价; int p[N]; /// Ai 矩阵的规模为 pi-1*pi ,两个相邻矩阵的相乘规模为 pi-1*pi*pi+1 int s[N][N]; ///记录 Ai - Aj中的最优分割点 void print_path(int i,int j){ if(i==j) { printf("A%d",i); }else{ printf("("); print_path(i,s[i][j]); print_path(s[i][j]+1,j); printf(")"); } } int main() { int n; scanf("%d",&n); for(int i=0;i<=n;i++){ scanf("%d",&p[i]); } for(int i=1;i<=n;i++) dp[i][i]=0; for(int l=2;l<=n;l++){ ///枚举长度 for(int i=1;i<=n-l+1;i++){ ///枚举起点 int j = i+l-1; ///终点 dp[i][j]=0xfffffff; for(int k = i;k<j;k++){ ///枚举决策点k if(dp[i][j]>dp[i][k]+dp[k+1][j]+p[i-1]*p[k]*p[j]){ dp[i][j]=dp[i][k]+dp[k+1][j]+p[i-1]*p[k]*p[j]; s[i][j] = k; } } } } printf("%d\n",dp[1][n]); print_path(1,n); } ///test: 6 /// 30 35 15 5 10 20 25 结果为 15125次
了解了矩阵链乘,接下来这道题的思想就和矩阵链乘一样了。
状态转移方程: if(i==j) dp[i][j]=1;
if(i<j) if(str[i]与str[j]匹配) dp[i][j] = dp[i+1][j-1]
else dp[i][j] = min(dp[i][k],dp[k+1][j]); k为断开的位置(决策点)
这题很坑,,用EOF结束死活过不了,看了一下Discuss区去掉EOF就AC
#include<string.h> #include<stdio.h> #include <iostream> using namespace std; #define N 105 char str[N]; int dp[N][N]; ///代表i-j区间最少添加的括号数 int s[N][N]; ///当i>j时,直接返回,不需要输出 ///当i==j时,s[i][j]为1,至少要加一个括号,如果str[i]为‘(‘ 或者‘)‘,输出"()",否则输出"[]" ///当i<j时,如果s[i][j]>=0,说明从i到j断开了,则递归调用print_path(i, s[i][j])和print_path(s[i][j]+1, j); ///如果s[i][j]==-1,说明没有断开,如果 则输出str[i],print_path(i+1, j-1); 和str[j] void print_path(int i,int j) { if(i>j) return ; else if(i==j) { if(str[i]==‘(‘||str[i]==‘)‘) printf("()"); else printf("[]"); } else { if(s[i][j]==-1) { printf("%c",str[i]); print_path(i+1,j-1); printf("%c",str[j]); } else { print_path(i,s[i][j]); print_path(s[i][j]+1,j); } } } int main() { scanf("%s",str+1); ///输入str+1 让字符串第0位为空,输入串占第 1-len位 int len = strlen(str+1); for(int i=1; i<=len; i++) dp[i][i]=1; for(int l = 2; l<=len; l++) { for(int i=1; i<=len-l+1; i++) { int j = i+l-1; if((str[i]==‘(‘&&str[j]==‘)‘)||(str[i]==‘[‘&&str[j]==‘]‘)) { dp[i][j] = dp[i+1][j-1]; s[i][j] = -1; } else dp[i][j]=0x7ffffff; for(int k=i; k<j; k++) ///寻找决策点 i<=k<j { if(dp[i][j]>dp[i][k]+dp[k+1][j]) { dp[i][j] = dp[i][k]+dp[k+1][j]; s[i][j] = k; } } } } //printf("%d\n",dp[1][len]); print_path(1,len); printf("\n"); return 0; }
标签:
原文地址:http://www.cnblogs.com/liyinggang/p/5374629.html