Educational Codeforces Round 165 (Rated for Div. 2) C. Minimizing the Sum题解
题意
Codeforces Round 809 (Div. 2) D1. Chopping Carrots (Easy Version)
给你两个整数, 一个数组 。你可以进行如下操作最多 次:选定一个数 ,让其变为相邻的数(变为中的一个), 与 不相邻。
输出数组各个元素之和最小值。
思路
这道题观察数据范围可以猜测解法可能与 有关,复杂度可能为 甚至 。
打假一下贪心做法,就是时刻维护相邻元素的差,每次操作都选取最大的差来操作。
例如如下数据:
4 2
120 70 1 80
如果按照贪心思路,最后数组变为了120 1 1 1
,但正确结果的数组应该是1 1 1 80
。
这道题需要提取出如下的性质:
- 如果一段区间被操作了,那么最终的数组一定是这个区间中的最小值
- 操作的区间与区间之间不会有交叉部分
这里就要用到dp来解决(一般我的各种算法都不行,贪心也被hack时候,就会考虑dp)。
设定 表示前 个元素中,用了 次操作后的元素之和的最小值。
根据定义,表示出状态转移: 。其中 表示 区间中的最小值
代码
#include <algorithm>
#include <cmath>
#include <cstring>
#include <iostream>
#include <map>
#include <queue>
#include <set>
#include <stack>
#include <string>
#include <vector>
#include <bitset>
#define fi first
#define se second
using namespace std;
using ll = long long;
using ull = unsigned long long;
using pii = pair<int, int>;
const ll inf = 0x3f3f3f3f3f3f3f3f;
const ull P = 131;
const double eps = 1e-4;
const int N = 3e5+10, M = 11;
ll n, k, a[N];
ll f[N][M];
void solve() {
scanf("%lld%lld", &n, &k);
for (int i=1; i<=n; i++)
scanf("%lld", &a[i]);
for (int i=0; i<=n; i++)
for (int j=0; j<=k+1; j++)
f[i][j] = inf;
f[0][0] = 0;
for (int i=1; i<=n; i++) {
for (int j=0; j<=k; j++) {
ll minv = a[i];
for (int d=0; d<=j && i-d-1>=0; d++) {
minv = min(minv, a[i-d]);
f[i][j] = min(f[i][j], f[i-d-1][j-d] + minv * (d + 1));
}
}
}
ll res = inf;
for (int i=0; i<=k; i++) res = min(res, f[n][i]);
printf("%lld\n", res);
}
int main() {
// multiple case
int t; scanf("%d", &t);
while(t--) {
solve();
}
// single case
// solve();
return 0;
}
不忘初心方得始终
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通