目录
一、题目
1、题目描述
2、输入输出
2.1输入
2.2输出
3、原题链接
二、解题报告
1、思路分析
2、复杂度
3、代码详解
一、题目
1、题目描述
2、输入输出
2.1输入
2.2输出
3、原题链接
865B - Ordering Pizza
二、解题报告
1、思路分析
如果我们不考虑披萨数目的限制,我们贪心的为每个人分配最优披萨,会有什么结果?
最多多买一个披萨,因为每个人要的披萨的片数是固定的
如果买多了,最多调整 部分人 1->2 或者 2->1
不会同时 1<->2,不然的话我们总能取消1->2 或者 2->1 来减少代价
那么对于两种方案,不管调谁,都去调代价最小的即可
2、复杂度
时间复杂度:O(NlogN) 空间复杂度:O(NlogN)
3、代码详解
#include <bits/stdc++.h>
// #define DEBUG
using u32 = unsigned;
using i64 = long long;
using u64 = unsigned long long;
constexpr int inf32 = 1E9 + 7;
constexpr i64 inf64 = 1E18 + 7;
void solve() {
int n, S;
std::cin >> n >> S;
std::vector<int> s(n), a(n), b(n);
i64 cnt1 = 0, cnt2 = 0;
i64 ans = 0;
std::vector<int> g1, g2;
for (int i = 0; i < n; ++ i) {
std::cin >> s[i] >> a[i] >> b[i];
if (a[i] > b[i]) {
g1.push_back(i);
cnt1 += s[i];
ans += 1LL * s[i] * a[i];
} else {
g2.push_back(i);
cnt2 += s[i];
ans += 1LL * s[i] * b[i];
}
}
if ((cnt1 + cnt2 + S - 1) / S == (cnt1 + S - 1) / S + (cnt2 + S - 1) / S) {
std::cout << ans << '\n';
return;
}
int to2 = cnt1 % S, to1 = cnt2 % S;
i64 wto2 = 0, wto1 = 0;
std::ranges::sort(g1, [&](int i, int j){
return a[i] - b[i] < a[j] - b[j];
});
for (int i : g1) {
int t = std::min(to2, s[i]);
to2 -= t;
wto2 += 1LL * t * (b[i] - a[i]);
}
std::ranges::sort(g2, [&](int i, int j){
return b[i] - a[i] < b[j] - a[j];
});
for (int i : g2) {
int t = std::min(to1, s[i]);
to1 -= t;
wto1 += 1LL * t * (a[i] - b[i]);
}
std::cout << ans + std::max(wto1, wto2) << '\n';
}
int main() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
#ifdef DEBUG
int cur = clock();
freopen("in.txt", "r", stdin);
freopen("out.txt", "w", stdout);
#endif
int t = 1;
// std::cin >> t;
while (t--) {
solve();
}
#ifdef DEBUG
std::cerr << "run-time: " << clock() - cur << '\n';
#endif
return 0;
}