最少交换次数

一、题目描述

给出数字K,请输出所有结果小于K的整数组合到一起的最少交换次数。

组合一起是指满足条件的数字相邻,不要求相邻后在数组中的位置。

二、输入输出描述

1. 输入描述

  • 第一行输入一个整数数组(-100 <= 数组中数值 <= 100),元素以空格分隔;
  • 第二行输入一个目标整数 K(-100 <= K <= 100)。

2. 输出描述

输出一个整数,表示将所有结果小于K的整数组合在一起的最少交换次数。

三、示例

|----|-------------------------------------------|
| 输入 | 1 3 1 4 0 2 |
| 输出 | 1 |
| 说明 | 小于2的表达式是1 1 0, 共三种可能将所有符合要求数字组合一起,最少交换1次。 |

四、解题思路

1. 核心思想

将 "让所有小于 k 的元素集中" 的问题,转化为 "找到长度为count(小于 k 的元素总数)的子数组,其中包含的'大于等于 k 的元素个数最少'"------ 因为这个最少个数就是该子数组需要交换的次数(用窗口外的小于 k 的元素替换窗口内的大于等于 k 的元素)。

2. 问题本质分析

问题本质是子数组筛选优化

  • 所有小于 k 的元素必须形成连续子数组,该子数组的长度固定为count(小于 k 的元素总数),无法改变。
  • 交换次数的本质:窗口内 "非目标元素"(≥k)的个数(需要用窗口外的 "目标元素"(<k)替换它们,每替换 1 个就是 1 次交换)。
  • 因此,问题转化为:在所有长度为count的子数组中,找到 "非目标元素个数最少" 的子数组,其个数即为最小交换次数。
3. 核心逻辑
  • 固定窗口长度:窗口长度 = 小于 k 的元素总数count(只有这个长度的窗口能容纳所有目标元素)。
  • 初始窗口计数:计算第一个窗口(前count个元素)的非目标元素个数(初始交换次数)。
  • 滑动窗口优化:通过滑动窗口动态更新窗口内的非目标元素个数,避免重复遍历(时间复杂度从 O (n²) 优化到 O (n)),最终找到最小值。
4. 步骤拆解
  1. 输入解析与预处理

    • 读取数组nums和目标值k
    • 统计数组中小于 k 的元素总数count
    • 特殊情况:若count == 1,直接输出 0(无需交换)。
  2. 计算初始窗口的交换次数

    • 遍历数组前count个元素,统计其中≥k 的元素个数,作为初始交换次数minSwapCount(同时赋值给临时变量tmpSwapCount用于滑动更新)。
  3. 滑动窗口动态更新

    • 遍历数组从索引count到末尾(窗口右边界):
      • 确定当前窗口的左边界(j - countj为当前右边界)。
      • 对比左边界元素和右边界元素的类型(是否 < k),更新tmpSwapCount
      • tmpSwapCount更新minSwapCount(保留最小值)。
  4. 输出结果

    • 输出minSwapCount,即最少交换次数。

五、代码实现

java 复制代码
import java.util.Scanner;

public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        // 使用Scanner读取一行输入
        String numsStr = scanner.nextLine();
        // 按空格分割字符串并转换为整数数组
        String[] numParts = numsStr.split(" ");
        int[] nums = new int[numParts.length];
        for (int i = 0; i < numParts.length; i++) {
            nums[i] = Integer.parseInt(numParts[i]);
        }
        // 读取一个整数k
        int k = scanner.nextInt();
        // 计算数组中小于k的元素数量
        int count = 0;
        for (int num : nums) {
            if (num < k) {
                count++;
            }
        }
        // 如果小于k的元素数量为1,直接输出0
        if (count == 1) {
            System.out.println(0);
            return;
        }
        // 计算最少交换次数
        int minSwapCount = 0;
        for (int i = 0; i < count; i++) {
            if (nums[i] >= k) {
                minSwapCount++;
            }
        }
        int tmpSwapCount = minSwapCount;
        // 使用滑动窗口更新最小交换次数
        for (int j = count; j < nums.length; j++) {
            int preLeft = j - count;
            int curRight = j;
            if (nums[preLeft] >= k && nums[curRight] < k) {
                tmpSwapCount--;
            } else if (nums[preLeft] < k && nums[curRight] >= k) {
                tmpSwapCount++;
            }
            minSwapCount = Math.min(minSwapCount, tmpSwapCount);
        }
        // 输出最终的最小交换次数
        System.out.println(minSwapCount);
    }
}
相关推荐
爱学英语的程序员1 分钟前
面试官:你了解过哪些数据库?
java·数据库·spring boot·sql·mysql·mybatis
xhbaitxl4 分钟前
算法学习day39-动态规划
学习·算法·动态规划
I_LPL5 分钟前
day23 代码随想录算法训练营 回溯专题2
算法·hot100·回溯算法·求职面试
智者知已应修善业7 分钟前
【洛谷P9975奶牛被病毒传染最少数量推导,导出多样例】2025-2-26
c语言·c++·经验分享·笔记·算法·推荐算法
猿小羽7 分钟前
AIGC 应用工程师(3-5 年)面试题精讲:从基础到实战的系统备战清单
面试·大模型·aigc·agent·rag
m0_7369191023 分钟前
C++中的委托构造函数
开发语言·c++·算法
小小小小王王王29 分钟前
洛谷-P1886 【模板】单调队列 / 滑动窗口
c++·算法
callJJ38 分钟前
Spring AI 文本聊天模型完全指南:ChatModel 与 ChatClient
java·大数据·人工智能·spring·spring ai·聊天模型
CBeann38 分钟前
企业级规则引擎落地实战:动态脚本引擎 QLExpress ,真香!
java·ai·大模型·规则引擎·qlexpress·大厂实战项目
懈尘38 分钟前
从 Java 1.7 到 Java 21:逐版本深入解析新特性与平台演进
java·开发语言