Skip to main content

Taking Maximum Energy From the Mystic Dungeon (LeetCode)

Problem statement:​

In a mystic dungeon, n magicians are standing in a line. Each magician has an attribute that gives you energy. Some magicians can give you negative energy, which means taking energy from you.

You have been cursed in such a way that after absorbing energy from magician i, you will be instantly transported to magician (i + k). This process will be repeated until you reach the magician where (i + k) does not exist.

In other words, you will choose a starting point and then teleport with k jumps until you reach the end of the magicians' sequence, absorbing all the energy during the journey.

You are given an array energy and an integer k. Return the maximum possible energy you can gain.

Example 1:

Input: energy = [5,2,-10,-5,1], k = 3 Output: 3

Explanation: We can gain a total energy of 3 by starting from magician 1 absorbing 2 + 1 = 3.

Example 2:

Input: energy = [-2,-3,-1], k = 2 Output: -1

Explanation: We can gain a total energy of -1 by starting from magician 2.

Constraints:

  • 1 <= energy.length <= 105
  • -1000 <= energy[i] <= 1000
  • 1 <= k <= energy.length - 1

Solutions:​

Approch 1:​

There will be k non-overlapping sets of integers that the energy array can be partitioned into. With adjacent elements in any given partition being k-index positions apart in the original input.

Additionally, since we're always moving from left to right, that means any energies that occur more to the right will need to be absorbed if we start to their left; while energies occuring towards the left may be skipped if we simply choose to start at an index position to the right of those. Thus keeping track of suffix-sums of partitioned energies, while iterating from right to left, will be our strategy to solving this.

code:​

Written by @Ajay-Dhangar
class Solution:
def maximumEnergy(self, energy: List[int], k: int) -> int:
res, n = -math.inf, len(energy)

# iterate through every possible starting position
for p in range(k):
acc_energy = 0

# iterate from the right to left, in k-steps
for i in range(n - p - 1, -1, -k):
acc_energy += energy[i]
res = max(res, acc_energy)

return res

Complexity​

Time complexity: O(kβˆ—n/k)O(kβˆ—n/k) = O(n)O(n)

Space complexity: O(1)O(1)

Approch 2:​

Intuition

  • Need to focus - you can start from any point and need to take all till end
  • We need to iterate from end and take the max of all intermidiate sum

Approach

  • Iterate i over 0 to k, and fix the start point as end - i
  • Again start iterate over array and keep traking the max value

code

Written by @Ajay-Dhangar
#include <vector>
#include <algorithm>
#include <climits>

class Solution {
public:
int maximumEnergy(std::vector<int>& energy, int k) {
int ans = INT_MIN;
for (int i = 0; i < k; ++i) {
for (int j = energy.size() - 1 - i, sum = 0; j >= 0; j -= k) {
sum += energy[j];
ans = std::max(ans, sum);
}
}
return ans;
}
};

Complexity​

Time complexity: O(N)O(N)

Space complexity: O(1)O(1)

Video lecture:​