LeetCode 3494.酿造药水需要的最少总时间:模拟(贪心)——一看就懂的描述
【LetMeFly】3494.酿造药水需要的最少总时间:模拟(贪心)——一看就懂的描述
力扣题目链接:https://leetcode.cn/problems/find-the-minimum-amount-of-time-to-brew-potions/
给你两个长度分别为 n
和 m
的整数数组 skill
和 mana
。
在一个实验室里,有 n
个巫师,他们必须按顺序酿造 m
个药水。每个药水的法力值为 mana[j]
,并且每个药水 必须 依次通过 所有 巫师处理,才能完成酿造。第 i
个巫师在第 j
个药水上处理需要的时间为 timeij = skill[i] * mana[j]
。
由于酿造过程非常精细,药水在当前巫师完成工作后 必须 立即传递给下一个巫师并开始处理。这意味着时间必须保持 同步,确保每个巫师在药水到达时 马上 开始工作。
返回酿造所有药水所需的 最短 总时间。
示例 1:
输入: skill = [1,5,2,4], mana = [5,1,4,2]
输出: 110
解释:
药水编号 | 开始时间 | 巫师 0 完成时间 | 巫师 1 完成时间 | 巫师 2 完成时间 | 巫师 3 完成时间 |
---|---|---|---|---|---|
0 | 0 | 5 | 30 | 40 | 60 |
1 | 52 | 53 | 58 | 60 | 64 |
2 | 54 | 58 | 78 | 86 | 102 |
3 | 86 | 88 | 98 | 102 | 110 |
举个例子,为什么巫师 0 不能在时间 t = 52
前开始处理第 1 个药水,假设巫师们在时间 t = 50
开始准备第 1 个药水。时间 t = 58
时,巫师 2 已经完成了第 1 个药水的处理,但巫师 3 直到时间 t = 60
仍在处理第 0 个药水,无法马上开始处理第 1个药水。
示例 2:
输入: skill = [1,1,1], mana = [1,1,1]
输出: 5
解释:
- 第 0 个药水的准备从时间
t = 0
开始,并在时间t = 3
完成。 - 第 1 个药水的准备从时间
t = 1
开始,并在时间t = 4
完成。 - 第 2 个药水的准备从时间
t = 2
开始,并在时间t = 5
完成。
示例 3:
输入: skill = [1,2,3,4], mana = [1,2]
输出: 21
提示:
n == skill.length
m == mana.length
1 <= n, m <= 5000
1 <= mana[i], skill[i] <= 5000
挺有意思的题。
解题方法:模拟
药水也要按顺序生产,巫师也要按顺序施法,所以没有策略优劣问题,只要在尽可能早的时间开启下一个毒药的生产流水线就好了。
什么是尽可能早的时间?好问题:
寻找一个时间满足:
- 从这个时间开始生产一瓶毒药,轮到每个巫师时,这个巫师都是空闲的;
- 没有一个比这个时间还早的时间可以满足第一条。
那么这个时间就是“尽可能早的时间”。
如何获得这个最早时间?两种思路:
思路一:计算需要delay多久才能开始生产下一瓶毒药
默认delay为0,计算当前delay下每个巫师接手这瓶毒药的时间。
假设巫师生产完上一瓶毒药时间为a,在b时刻接手这瓶毒药,那么就说明第一个巫师需要多delay b−ab-ab−a的时间再开始生产。
思路二:计算最后一个巫师完成时间并向前反推
一个巫师开始生产一瓶毒药的时间为max(上一个巫师的完成时间,这个巫师上瓶毒药的完成时间)max(上一个巫师的完成时间,这个巫师上瓶毒药的完成时间)max(上一个巫师的完成时间,这个巫师上瓶毒药的完成时间)。
计算出最后一个巫师的完成时间后,向前减去每个巫师需要消耗的时间即可得到每个巫师的开始时间。
- 时间复杂度O(mn)O(mn)O(mn)
- 空间复杂度O(n)O(n)O(n)
AC代码
C++ - 倒推的方式
/** @LastEditTime: 2025-10-09 23:01:17*/
typedef long long ll;
class Solution {
public:ll minTime(vector<int>& skill, vector<int>& mana) {vector<ll> times(skill.size());for (int m : mana) {ll nowFinish = times[0] + skill[0] * m;for (int i = 1; i < skill.size(); i++) {nowFinish = max(nowFinish, times[i]) + skill[i] * m;}times.back() = nowFinish;for (int i = skill.size() - 2; i >= 0; i--) {times[i] = times[i + 1] - skill[i + 1] * m;}}return times.back();}
};
C++ - delay的方式
/** @LastEditTime: 2025-10-09 22:49:27*/
typedef long long ll;
class Solution {
public:ll minTime(vector<int>& skill, vector<int>& mana) {vector<ll> times(skill.size());for (int m : mana) {ll delay = 0;ll nowFinish = times[0] + skill[0] * m;for (int i = 1; i < skill.size(); i++) {delay += max(0ll, times[i] - nowFinish);nowFinish = max(nowFinish, times[i]) + skill[i] * m;}times[0] += delay + skill[0] * m;for (int i = 1; i < skill.size(); i++) {times[i] = times[i - 1] + skill[i] * m;}// cout << times[0] << endl; // *****}return times.back();}
};#if defined(_WIN32) || defined(__APPLE__)
/*
[1,5,2,4]
[5,1,4,2]110巫师0完成时间:
5
53
58
88
*/
int main() {string a, b;while (cin >> a >> b) {vector<int> va = stringToVector(a), vb = stringToVector(b);Solution sol;cout << sol.minTime(va, vb) << endl;}return 0;
}
#endif
Python - 倒推的方式
'''
LastEditTime: 2025-10-09 23:11:26
'''
from typing import Listclass Solution:def minTime(self, skill: List[int], mana: List[int]) -> int:n = len(skill)times = [0] * nfor m in mana:lastTime = times[0] + skill[0] * mfor i in range(1, n):lastTime = max(lastTime, times[i]) + skill[i] * mtimes[-1] = lastTimefor i in range(n - 2, -1, -1):times[i] = times[i + 1] - skill[i + 1] * mreturn times[-1]
Java - 倒推的方式
/** @LastEditTime: 2025-10-09 23:34:16*/
class Solution {public long minTime(int[] skill, int[] mana) {int n = skill.length;long[] times = new long[n];for (int m : mana) {long last = times[0] + skill[0] * m;for (int i = 1; i < n; i++) {last = Math.max(last, times[i]) + skill[i] * m;}times[n-1] = last;for (int i = n - 2; i >= 0; i--) {times[i] = times[i + 1] - skill[i + 1] * m;}}return times[n-1];}
}
Go - 倒推的方式
/** @LastEditTime: 2025-10-09 23:32:19*/
package mainfunc minTime(skill []int, mana []int) int64 {n := len(skill)times := make([]int64, n)for _, m := range mana {lastTime := times[0] + int64(skill[0] * m)for i := 1; i < n; i++ {lastTime = max3494(times[i], lastTime) + int64(skill[i] * m)}times[n-1] = lastTimefor i := n - 2; i >= 0; i-- {times[i] = times[i + 1] - int64(skill[i + 1] * m)}}return times[n-1]
}func max3494(a, b int64) int64 {if a > b {return a}return b
}
Rust - 倒推的方式
/** @LastEditTime: 2025-10-09 23:18:22*/
impl Solution {pub fn min_time(skill: Vec<i32>, mana: Vec<i32>) -> i64 {let n: usize = skill.len();let mut times: Vec<i64> = vec![0; n];for m in mana {let mut last_time: i64 = times[0] + (skill[0] * m) as i64;for i in 1..n {last_time = last_time.max(times[i]) + (skill[i] * m) as i64;}times[n-1] = last_time;for i in (0..n-1).rev() {times[i] = times[i + 1] - (skill[i + 1] * m) as i64;}}times[n-1]}
}
同步发文于CSDN和我的个人博客,原创不易,转载经作者同意后请附上原文链接哦~
千篇源码题解已开源