端点石子移动问题

问题描述

小S正在玩一个关于石子的游戏,游戏中给定了一些石子,它们位于一维数轴上的不同位置,位置用数组 stones 表示。如果某个石子处于最小或最大的一个位置,我们称其为端点石子。

在每个回合,小S可以将一颗端点石子移动到一个未占用的位置,使其不再是端点石子。值得注意的是,如果石子的位置是连续的,则游戏结束,因为没有可以进行的移动操作。

你需要帮助小S找到可以移动的最小次数。


测试样例

样例1:

输入:stones = [7, 4, 9]

输出:1

样例2:

输入:stones = [6, 5, 4, 3, 10]

输出:2

样例3:

输入:stones = [1, 2, 3, 4, 5]

输出:0

Java代码

ini 复制代码
 import java.util.Arrays;
  public class Main {
  public static int solution(int[] stones) {
    if (stones.length == 1) {
        return 0;
    }
    
    Arrays.sort(stones);
    int n = stones.length;
    // 特殊处理 n = 2 的情况
    if (n == 2) {
        if (stones[1] - stones[0] == 1) {
            return 0; // 相邻则不需要移动
        }
        return 2; // 不相邻则需要移动两次
    }

    // 计算最大移动次数
    int maxMoves = stones[n - 1] - stones[0] + 1 - n;

    // 计算最小移动次数,使用滑动窗口
    int minMoves = Integer.MAX_VALUE;
    int j = 0;
    for (int i = 0; i < n; i++) {
        // 确保窗口内最多有 n 个石子
        while (j < n && stones[j] - stones[i] + 1 <= n) {
            j++;
        }
        // 如果窗口内有 n - 1 个石子,且空位为 1,则需要特殊处理
        int alreadyInPlace = j - i;
        if (alreadyInPlace == n - 1 && stones[j - 1] - stones[i] + 1 == n - 1) {
            minMoves = Math.min(minMoves, 2);
        } else {
            minMoves = Math.min(minMoves, n - alreadyInPlace);
        }
    }

    return Math.min(minMoves, maxMoves);
}

public static void main(String[] args) {
    System.out.println(solution(new int[]{7, 4, 9}) == 1); // 输出:true
    System.out.println(solution(new int[]{6, 5, 4, 3, 10}) == 2); // 输出:true
    System.out.println(solution(new int[]{1, 2, 3, 4, 5}) == 0); // 输出:true
    System.out.println(solution(new int[]{6, 8}) == 1); // 输出:true
    System.out.println(solution(new int[]{11, 14}) == 2); // 输出:true
}
}

时间复杂度分析

  1. 排序操作

    • Arrays.sort(stones); 使用的是快速排序,时间复杂度为 O(n log n)
  2. 特殊处理 n = 2 的情况

    • 这部分是常数时间操作,时间复杂度为 O(1)
  3. 计算最大移动次数

    • int maxMoves = stones[n - 1] - stones[0] + 1 - n; 是常数时间操作,时间复杂度为 O(1)
  4. 计算最小移动次数,使用滑动窗口

    • 外层 for 循环遍历数组,时间复杂度为 O(n)
    • 内层 while 循环在最坏情况下会遍历整个数组,但由于 j 是递增的,每个元素最多被访问两次(一次在 for 循环中,一次在 while 循环中),所以内层循环的总时间复杂度为 O(n)
    • 因此,滑动窗口部分的总时间复杂度为 O(n)

总时间复杂度

  • 排序操作:O(n log n)
  • 特殊处理和滑动窗口:O(n)

综合起来,代码的总时间复杂度为 O(n log n),其中 n 是数组 stones 的长度。

相关推荐
leo__5202 小时前
基于两步成像算法的聚束模式SAR MATLAB实现
开发语言·算法·matlab
前端小白在前进2 小时前
力扣刷题:在排序数组中查找元素的第一个和最后一个位置
数据结构·算法·leetcode
某林2123 小时前
基于SLAM Toolbox的移动机器人激光建图算法原理与工程实现
stm32·嵌入式硬件·算法·slam
修炼地4 小时前
代码随想录算法训练营第四十三天 | 图论理论基础、深搜理论基础、卡码网98. 所有可达路径、797. 所有可能的路径、广搜理论基础
算法·深度优先·图论
iAkuya4 小时前
(leetcode)力扣100 23反转链表(迭代||递归)
算法·leetcode·链表
剪一朵云爱着4 小时前
PAT 1095 Cars on Campus
算法·pat考试
MicroTech20255 小时前
激光点云快速配准算法创新突破,MLGO微算法科技发布革命性点云配准算法技术
人工智能·科技·算法
Cathy Bryant5 小时前
傅里叶变换(一):简介
笔记·算法·数学建模·信息与通信·傅里叶分析
allan bull5 小时前
在节日中寻找平衡:圣诞的欢乐与传统节日的温情
人工智能·学习·算法·职场和发展·生活·求职招聘·节日
似水এ᭄往昔6 小时前
【C++】--封装红⿊树实现mymap和myset
开发语言·数据结构·c++·算法·stl