蓝桥杯每日一题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;
}
相关推荐
MATLAB代码顾问4 分钟前
混合粒子群-模拟退火算法(HPSO-SA)求解作业车间调度问题——附MATLAB代码
算法·matlab·模拟退火算法
Felven8 分钟前
C. Prefix Min and Suffix Max
算法
加农炮手Jinx8 分钟前
LeetCode 26. Remove Duplicates from Sorted Array 题解
算法·leetcode·力扣
加农炮手Jinx9 分钟前
LeetCode 88. Merge Sorted Array 题解
算法·leetcode·力扣
格林威9 分钟前
线阵工业相机:如何计算线阵相机的行频(Line Rate)?公式+实例
开发语言·人工智能·数码相机·算法·计算机视觉·工业相机·线阵相机
yueyue54312 分钟前
透过现象看本质:以fast_lio架构的整套算法的局部避障改为TEB算法为例深度探讨——如何成为一个合格的算法架构师?
算法·架构
梨花爱跨境12 分钟前
红人视频×A10算法:亚马逊转化率与流量闭环实战
算法
阿Y加油吧17 分钟前
二刷 LeetCode:75. 颜色分类 & 31. 下一个排列 复盘笔记
笔记·算法·leetcode
风筝在晴天搁浅18 分钟前
LeetCode 378.有序矩阵中第K小的元素
算法·矩阵
qeen8743 分钟前
【算法笔记】简单贪心
c++·笔记·算法·贪心算法