P2308 添加括号,区间dp,dfs过程展示

P2308 添加括号 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)

题目背景

给定一个正整数序列a(1),a(2),...,a(n),(1<=n<=20)

不改变序列中每个元素在序列中的位置,把它们相加,并用括号记每次加法所得的和,称为中间和。

例如:

给出序列是4,1,2,3。

第一种添括号方法:

((4+1)+(2+3))=((5)+(5))=(10)

有三个中间和是5,5,10,它们之和为:5+5+10=20

第二种添括号方法

(4+((1+2)+3))=(4+((3)+3))=(4+(6))=(10)

中间和是3,6,10,它们之和为19。

题目描述

现在要添上n-1对括号,加法运算依括号顺序进行,得到n-1个中间和,求出使中间和之和最小的添括号方法。

输入格式

共两行。 第一行,为整数n。(1< =n< =20) 第二行,为a(1),a(2),...,a(n)这n个正整数,每个数字不超过100。

输出格式

输出3行。 第一行,为添加括号的方法。 第二行,为最终的中间和之和。 第三行,为n-1个中间和,按照从里到外,从左到右的顺序输出。

输入输出样例

输入 #1复制

4
4 1 2 3

输出 #1复制

(4+((1+2)+3))
19
3 6 10

说明/提示

范围在题目上有说明。

解析:

这道题实际上是石子合并的升级版本,增加了过程展示这一不。

对于过程的展示,我没呢可以用dfs从最终答案往回搜索的到,应为最后的最优解一定是由前一个状态转移而来的,所以我们dfs最优解的状态即可。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<string>
#include<cstring>
#include<cmath>
#include<ctime>
#include<algorithm>
#include<utility>
#include<stack>
#include<queue>
#include<vector>
#include<set>
#include<map>
using namespace std;
typedef long long LL;
const int N = 100;
int n;
int f[N][N], a[N], sum[N], flg[N][N];

void print(int l, int r) {
	if (l == r) {
		printf("%d", a[l]);
		return;
	}
	printf("(");
	print(l, flg[l][r]);
	printf("+");
	print(flg[l][r] + 1, r);
	printf(")");
}

int F(int l, int r) {
	if (l == r) {
		return a[l];
	}
	int ret = F(l, flg[l][r]) + F(flg[l][r] + 1, r);
	printf("%d ", ret);
	return ret;
}

int main() {
	scanf("%d", &n);
	for (int i = 1; i <= n; i++) {
		scanf("%d", &a[i]);
		sum[i] = sum[i - 1] + a[i];
	}
	for (int l = 2; l <= n; l++) {
		for (int i = 1; i + l - 1 <= n; i++) {
			int j = i + l - 1;
			f[i][j] = 0x3f3f3f3f;
			for (int k = i; k + 1 <= j; k++) {
				if (f[i][j] > f[i][k] + f[k + 1][j] + sum[j] - sum[i - 1]) {
					f[i][j] = f[i][k] + f[k + 1][j] + sum[j] - sum[i - 1];
					flg[i][j] = k;
				}
			}
		}
	}
	print(1, n);
	printf("\n%d\n", f[1][n]);
	F(1, n);
	return 0;
}

posted @ 2023-10-25 16:45  Landnig_on_Mars  阅读(6)  评论(0编辑  收藏  举报  来源