Skip to content

Latest commit

 

History

History

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 

README.md

You are given a 0-indexed binary string s of length n on which you can apply two types of operations:

  • Choose an index i and invert all characters from index 0 to index i (both inclusive), with a cost of i + 1
  • Choose an index i and invert all characters from index i to index n - 1 (both inclusive), with a cost of n - i

Return the minimum cost to make all characters of the string equal.

Invert a character means if its value is '0' it becomes '1' and vice-versa.

 

Example 1:

Input: s = "0011"
Output: 2
Explanation: Apply the second operation with i = 2 to obtain s = "0000" for a cost of 2. It can be shown that 2 is the minimum cost to make all characters equal.

Example 2:

Input: s = "010101"
Output: 9
Explanation: Apply the first operation with i = 2 to obtain s = "101101" for a cost of 3.
Apply the first operation with i = 1 to obtain s = "011101" for a cost of 2. 
Apply the first operation with i = 0 to obtain s = "111101" for a cost of 1. 
Apply the second operation with i = 4 to obtain s = "111110" for a cost of 2.
Apply the second operation with i = 5 to obtain s = "111111" for a cost of 1. 
The total cost to make all characters equal is 9. It can be shown that 9 is the minimum cost to make all characters equal.

 

Constraints:

  • 1 <= s.length == n <= 105
  • s[i] is either '0' or '1'

Related Topics:
String, Dynamic Programming, Greedy

Similar Questions:

Hints:

  • For every index i, calculate the number of operations required to make the prefix [0, i - 1] equal to the character at index i, denoted prefix[i].
  • For every index i, calculate the number of operations required to make the suffix [i + 1, n - 1] equal to the character at index i, denoted suffix[i].
  • The final string will contain at least one character that is left unchanged; Therefore, the answer is the minimum of prefix[i] + suffix[i] for every i in [0, n - 1].

Solution 1. DP

Let left[0/1][i+1] be the cost of turning all s[0..i] to 0/1, and right[0/1][i] be the cost of turning all s[i..N-1] to 0/1. The answer is the minimum of all left[0][i] + right[0][i], left[1][i] + right[1][i] (0 <= i <= N).

For left[0/1][i+1]:

// If `s[i] == 0`
left[0][i+1] = min(left[0][i], // turn s[0..i-1] to 0
                    left[1][i] + i) // turn s[0..i-1] to 1, then flip s[0..i-1] to 1 with cost i
left[1][i+1] = min(left[0][i] + i + 1, // turn s[0..i-1] to 0, then flip s[0..i] to 1 with cost i+1
                    left[1][i] + 2*i + 1) // turn s[0..i-1] to 1, then flip s[0..i-1] to 0 with cost i, then flip s[0..i] to 1 with cost i+1
             = min(left[0][i], left[1][i] + i) + i + 1
// If `s[i] == 1`
left[0][i+1] = min(left[0][i] + 2*i + 1, // turn s[0..i-1] to 0, then flip s[0..i-1] to 1 with cost i, then flip s[0..i] to 0 with cost i+1
                    left[1][i] + i + 1) // turn s[0..i-1] to 1, then flip s[0..i] with cost i + 1
             = min(left[0][i] + i, left[1][i]) + i + 1
left[1][i+1] = min(left[0][i] + i, // turn s[0..i-1] to 0, then flip s[0..i-1] to 1 with cost i
                    left[1][i]) // turn s[0..i-1] to 1

Similarly

if s[i] == 0
right[0][i] = min(right[0][i+1], right[1][i+1] + N - i - 1)
right[1][i] = min(right[0][i+1], right[1][i+1] + N - i - 1) + N - i
if s[i] == 1
right[0][i] = min(right[0][i+1] + N - i - 1, right[1][i+1]) + N - i
right[1][i] = min(right[0][i+1] + N - i - 1, right[1][i+1])
// OJ: https://leetcode.com/problems/minimum-cost-to-make-all-characters-equal
// Author: github.com/lzl124631x
// Time: O(N)
// Space: O(N)
class Solution {
public:
    long long minimumCost(string s) {
        long long N = s.size(), ans = LLONG_MAX;
        vector<vector<long long>> right(2, vector<long long>(N + 1)), left(2, vector<long long>(N + 1));
        for (int i = N - 1; i >= 0; --i) {
            if (s[i] == '0') {
                long long val = min(right[0][i + 1], right[1][i + 1] + N - i - 1);
                right[0][i] = val;
                right[1][i] = val + N - i;
            } else {
                long long val = min(right[0][i + 1] + N - i - 1, right[1][i + 1]);
                right[0][i] = val + N - i;
                right[1][i] = val;
            }
        }
        for (int i = 0; i < N; ++i) {
            if (s[i] == '0') {
                long long val = min(left[0][i], left[1][i] + i);
                left[0][i + 1] = val;
                left[1][i + 1] = val + i + 1;
            } else {
                long long val = min(left[0][i] + i, left[1][i]);
                left[0][i + 1] = val + i + 1;
                left[1][i + 1] = val;
            }
        }
        for (int i = 0; i <= N; ++i) {
            ans = min({ ans, left[0][i] + right[0][i], left[1][i] + right[1][i] });
        }
        return ans;
    }
};

Solution 2. Greedy

We scan from left to right. Whenever we see two characters (s[i-1] and s[i]) that are different, we pick the minimum between these two options:

  • Flip s[0..i-1] with cost i
  • Flip s[i..N-1] with cost N-i

This algorithm works because the order in which we do the flips doesn't matter. Let's say we made the choice at s[0] vs s[1], and now we look at s[1] vs s[2], no matter which choice we make, s[0] and s[1] will be flipped together.

// OJ: https://leetcode.com/problems/minimum-cost-to-make-all-characters-equal
// Author: github.com/lzl124631x
// Time: O(N)
// Space: O(1)
// Ref: https://leetcode.com/problems/minimum-cost-to-make-all-characters-equal/solutions/3570185/single-pass-iterative-easy-explanation-think-greedy-one-line-clean-code-c/
class Solution {
public:
    long long minimumCost(string s) {
        long long N = s.size(), ans = 0;
        for (int i = 1; i < N; ++i) {
            if (s[i] != s[i - 1]) ans += min((long long)i, N - i);
        }
        return ans;
    }
};