蓝桥杯每日一题2023.10.5

3420. 括号序列 - AcWing题库

题目描述

题目分析

对于这一我们需要有前缀知识完全背包

完全背包的朴素写法:

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
const int N = 1010;
int n, m, v[N], w[N], f[N][N];
int main()
{
	cin >> n >> m;
	for(int i = 1; i <= n; i ++)cin >> v[i] >> w[i];
	for(int i = 1; i <= n; i ++)
	{
		for(int j = 0; j <= m; j ++)
		{
			for(int k = 0; k * v[i] <= j; k ++)
			{
				f[i][j] = max(f[i][j], f[i - 1][j - v[i] * k] + w[i] * k);
			}
		}
	}
	cout << f[n][m] << '\n';
	return 0;
}

经行优化:

f[i][j] = f[i - 1][j - v[i] * k] + w[i] * k

f[i][j] = max(f[i - 1][j], f[i - 1][j - v] + w, f[i - 1][j - 2v] + 2w, f[i - 1][j - 3v] + 3w, ...)

f[i][j - v] = max( f[i - 1][j - v], f[i - 1][j - 2v] + w, f[i - 1][j - 3v] + 2w, ...)

f[i][j] = max(f[i - 1][j], f[i][j - v] + w)

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
const int N = 1010;
int n, m, v[N], w[N], f[N][N];
int main()
{
	cin >> n >> m;
	for(int i = 1; i <= n; i ++)cin >> v[i] >> w[i];
	for(int i = 1; i <= n; i ++)
	{
		for(int j = 0; j <= m; j ++)
		{
			for(int k = 0; k * v[i] <= j; k ++)
			{
				f[i][j] = f[i - 1][j];
				if(j >= v[i])f[i][j] = max(f[i][j], f[i][j - v[i]] + w[i]);
			}
		}
	}
	cout << f[n][m] << '\n';
	return 0;
}

首先由题意知我们左右括号的数量必须相等,对于任意前缀的左括号的数量必须大于等于有括号的数量(如果小于则此处必定需要添加括号)

我们可以分为两种方案使其独立存在,一种是只添加左括号,一种是只添加右括号,这两种方案各进行一次,将方案数相乘则为总方案数,对于左右进行的操作只需用同一代码即可,我们可以只写对左括号进行操作,对于右括号操作我们只需要将字符串翻转即可实现操作

使用动态规划来记录方案数

f[i][j] :只考虑前i部分,左括号比右括号多j 个的所有方案的集合(不同数量的左括号的方案数)

1.若s[i] == '(' f[i][j] = f[i - 1][j - 1](考虑前i - 1部分时,左括号数量比右括号数量多j - 1个,那么第i部分左括号就比右括号多j个)

2.若s[i] == ')' f[i][j] = f[i - 1][j + 1] + f[i - 1][j] + ... + f[i - 1][0](考虑前i - 1部分左括号数量最多比右括号数量多j + 1个,才能在第i部分通过添加或者不加左括号使左括号的数量比右括号的数量多j个)注:这里类似于完全背包的优化:f[i][j] = f[i - 1][j + 1] + f[i][j - 1],考虑越界问题,f[i][0]特判(j == 0,j - 1 = -1越界)f[i][0]可以考虑前i - 1部分左括号数和右括号数相等 和 左括号数比右括号数多一个的和

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 5010, mod = 1e9 + 7;
char s[N];
int n;
ll f[N][N];
ll work()
{
	memset(f, 0, sizeof f);
	f[0][0] = 1;
	for(int i = 1; i <= n; i ++)
	{
		if(s[i] == '(')
		{
			for(int j = 1; j <= n; j ++)f[i][j] = f[i - 1][j - 1];
		}
		else
		{
			f[i][0] = (f[i - 1][0] + f[i - 1][1]) % mod;
			for(int j = 1; j <= n; j ++)f[i][j] = (f[i - 1][j + 1] + f[i][j - 1]) % mod;
		}
	}
	for(int i = 0; i <= n; i ++)
	{
		if(f[n][i])return f[n][i];
	}
	return -1;
}
int main()
{
	cin >> s + 1;
	n = strlen(s + 1);
	ll l = work();
	reverse(s + 1, s + n + 1);
	for(int i = 1; i <= n; i ++)
	{
		if(s[i] == '(')s[i] = ')';
		else s[i] = '(';
	}
	ll r = work();
	cout << l * r % mod;
	return 0;
}
相关推荐
Kent_J_Truman11 分钟前
【平方差 / C】
算法
一直学习永不止步12 分钟前
LeetCode题练习与总结:H 指数--274
java·数据结构·算法·leetcode·数组·排序·计数排序
Amor风信子38 分钟前
华为OD机试真题---跳房子II
java·数据结构·算法
戊子仲秋1 小时前
【LeetCode】每日一题 2024_10_2 准时到达的列车最小时速(二分答案)
算法·leetcode·职场和发展
邓校长的编程课堂1 小时前
助力信息学奥赛-VisuAlgo:提升编程与算法学习的可视化工具
学习·算法
sp_fyf_20241 小时前
计算机前沿技术-人工智能算法-大语言模型-最新研究进展-2024-10-03
人工智能·算法·机器学习·计算机视觉·语言模型·自然语言处理
Eric.Lee20212 小时前
数据集-目标检测系列- 螃蟹 检测数据集 crab >> DataBall
python·深度学习·算法·目标检测·计算机视觉·数据集·螃蟹检测
林辞忧2 小时前
算法修炼之路之滑动窗口
算法
￴ㅤ￴￴ㅤ9527超级帅2 小时前
LeetCode hot100---二叉树专题(C++语言)
c++·算法·leetcode
liuyang-neu2 小时前
力扣 简单 110.平衡二叉树
java·算法·leetcode·深度优先