【LeetCode】1769.移动所有球到每个盒子所需的最小操作数

题目描述

有 n 个盒子。给你一个长度为 n 的二进制字符串 boxes ,其中 boxes[i] 的值为 ‘0’ 表示第 i 个盒子是 空 的,而 boxes[i] 的值为 ‘1’ 表示盒子里有 一个 小球。
在一步操作中,你可以将 一个 小球从某个盒子移动到一个与之相邻的盒子中。第 i 个盒子和第 j 个盒子相邻需满足 abs(i - j) == 1 。注意,操作执行后,某些盒子中可能会存在不止一个小球。
返回一个长度为 n 的数组 answer ,其中 answer[i] 是将所有小球移动到第 i 个盒子所需的 最小 操作数。
每个 answer[i] 都需要根据盒子的 初始状态 进行计算。

示例 1:

输入:boxes = “110”
输出:[1,1,3]
解释:每个盒子对应的最小操作数如下:

  • 第 1 个盒子:将一个小球从第 2 个盒子移动到第 1 个盒子,需要 1 步操作。
  • 第 2 个盒子:将一个小球从第 1 个盒子移动到第 2 个盒子,需要 1 步操作。
  • 第 3 个盒子:将一个小球从第 1 个盒子移动到第 3 个盒子,需要 2 步操作。将一个小球从第 2 个盒子移动到第 3 个盒子,需要 1 步操作。共计 3 步操作。

示例 2:

输入:boxes = “001011”
输出:[11,8,5,4,3,4]

提示:

n == boxes.length
1 <= n <= 2000
boxes[i] 为 ‘0’ 或 ‘1’

方法一:两次遍历

class Solution {
public:
    vector<int> minOperations(string boxes) {
        int n = boxes.size();
        vector<int> flag;
        vector<int> answer(n);
        // 一次遍历:保存有小球的下标
        for(int i=0; i<n; i++){
            if(boxes[i] == '1') flag.push_back(i);
        }
        
        // 二次遍历:小球下标和当前下标的差值就是操作次数
        for(int i=0; i<n; i++){
            for(int j=0; j<flag.size(); j++){
                answer[i] += abs(flag[j] - i);
            }
        }
    return answer;
    }
};

方法二:方法一的优化-一次遍历
没必要保存小球下标的位置,直接计算即可

class Solution {
public:
    vector<int> minOperations(string boxes) {
        int n = boxes.size();
        vector<int> answer(n);
        // 一次遍历
        for(int i=0; i<n; i++){
        	// 外层遍历移动目标盒子
            for(int j=0; j<n; j++){
            	// 内层遍历盒子里是否有小球
                if(boxes[j] == '1') answer[i] += abs(j - i);
            }
        }
    return answer;
    }
};

方法三:方法二的优化,减少时间
首先考虑该盒子中是否有小球,如果有再继续考虑,没有的话直接遍历下一个小球

class Solution {
public:
    vector<int> minOperations(string boxes) {
        int n = boxes.size();
        vector<int> answer(n);
        for(int i=0; i<n; i++){
        	// 外层遍历是否有小球需要移动
            if(boxes[i] == '0') continue;
            for(int j=0; j<n; j++){
            // 内层遍历要移动目标盒子
                answer[j] += abs(j - i);
            }
        }
    return answer;
    }
};

心得
这道题不算很难,一开始先用 动态规划 去做,但是我还是不太熟练,不太会去找临界条件,觉得有些麻烦,最后还是用最普通的思路,以下做介绍。

方法一:双重循环模拟

  • 思路
    • 问题在于计算将 所有小球 移动到第 i 个盒子所需的 最小 操作数,并且每次只能移动 一个 小球。从第二个例子可以发现,第 j 个盒子里的小球移动到 第 i 个盒子,所需的操作数就是 abs(j - i) ,这里使用了 abs 是考虑到可能是将左边的小球移动到右边的盒子,此时 j - i 为负数。
    • 确定好 「abs(j - i)」之后,需要知道 j 的位置,因此我首先遍历了 boxes 数组,将有小球的盒子下标保存在 flag 数组里。
    • 之后只需要双重模拟循环,依次将每个盒子 i 与 flag 中有小球的盒子下标相减,就可以得到操作次数。
  • 时间复杂度: O(n2
  • 空间复杂度: O(n)
    【LeetCode】1769.移动所有球到每个盒子所需的最小操作数_第1张图片

方法二:双重循环模拟(方法一的优化)

  • 思路
    • 总体思路和方法一差不多,考虑到 flag数组 有点多余,因为可以在双重循环的时候顺便判断该盒子里是否有小球,即 「 if ( boxes[j] == ‘1’ ) 」,因此方法二去掉了第一次遍历,空间复杂度降低了。
  • 时间复杂度: O(n2
  • 空间复杂度: O(n)
    【LeetCode】1769.移动所有球到每个盒子所需的最小操作数_第2张图片

方法三:双重循环模拟(方法二的优化)

  • 思路
  • 总体思路和方法二差不多,将内外层循环的定义做了调换,首先考虑遍历到的盒子里是否有小球,如果有的话再把它加入到移动目标盒子中。除了极端情况(每个盒子都有小球),这样子的遍历次数肯定会少于 n2,运行时间提高了一倍。
  • 时间复杂度 < O(n2
  • 空间复杂度: O(n)
    【LeetCode】1769.移动所有球到每个盒子所需的最小操作数_第3张图片

你可能感兴趣的:(LeetCode刷题,leetcode,算法,动态规划)