LeetCode 算法:轮转数组c++

2024-06-10 04:52

本文主要是介绍LeetCode 算法:轮转数组c++,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原题链接🔗:轮转数组
难度:中等⭐️⭐️

题目

给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数。

示例 1:
输入: nums = [1,2,3,4,5,6,7], k = 3
输出: [5,6,7,1,2,3,4]
解释:
向右轮转 1 步: [7,1,2,3,4,5,6]
向右轮转 2 步: [6,7,1,2,3,4,5]
向右轮转 3 步: [5,6,7,1,2,3,4]

示例 2:
输入:nums = [-1,-100,3,99], k = 2
输出:[3,99,-1,-100]
解释:
向右轮转 1 步: [99,-1,-100,3]
向右轮转 2 步: [3,99,-1,-100]

提示

  • 1 <= nums.length <= 105
  • -231 <= nums[i] <= 231 - 1
  • 0 <= k <= 105
    进阶
  • 尽可能想出更多的解决方案,至少有 三种 不同的方法可以解决这个问题。
  • 你可以使用空间复杂度为 O(1) 的 原地 算法解决这个问题吗?

题解

原地轮转数组

  1. 题解

这种方法通过三次反转来实现原地轮转,不需要额外的空间。

步骤:

  • 反转整个数组。
  • 反转前 k 个元素。
  • 反转 k 个元素后的剩余部分。
  1. 复杂度:时间复杂度O(n),空间复杂度 O(n)。
  2. 过程
  1. 头文件和命名空间:

    • #include <vector>: 包含标准库中的 vector 容器。
    • #include <iostream>: 包含输入输出流。
    • using namespace std;: 使用标准命名空间,这样我们就不需要在标准库类型和函数前加 std:: 前缀。
  2. 辅助函数 reverse:

    • 函数原型:void reverse(vector<int>& nums, int start, int end)
    • 功能:反转 nums 数组中从索引 start 到索引 end 的部分。
    • 实现:使用 while 循环和 swap 函数来交换 startend 指向的元素,然后 start 向前移动,end 向后移动,直到 start 大于或等于 end
  3. Solution:

    • 包含一个公有成员函数 rotate,用于实现数组的轮转。
    • 输入参数:nums 是要轮转的数组,k 是轮转的步数。
    • 实现逻辑:
      • 首先检查 nums 是否为空或者 k 是否是数组长度的倍数,如果是,则直接返回,因为不需要做任何操作。
      • 然后,通过三次反转实现数组的轮转:
        • 反转整个数组。
        • 反转数组的前 k 个元素。
        • 反转数组从索引 k 到末尾的部分。
  4. 主函数 main:

    • 创建 Solution 类的实例。
    • 定义一个 vector<int> 类型的数组 nums,初始化为 {1, 2, 3, 4, 5, 6, 7}
    • 定义一个整数 k,值为 3,表示要将数组向右轮转 3 步。
    • 打印原始数组。
    • 调用 rotate 方法对数组进行轮转。
    • 打印轮转后的数组。
  5. 输出:

    • 程序首先打印原始数组。
    • 然后调用 rotate 方法,将数组 [1, 2, 3, 4, 5, 6, 7] 向右轮转 3 步,结果应该是 [5, 6, 7, 1, 2, 3, 4]
    • 最后打印轮转后的数组。
  1. c++ demo
#include <vector>
#include <iostream>using namespace std;// 辅助函数,用于反转数组中从 start 到 end 的部分
void reverse(vector<int>& nums, int start, int end) {while (start < end) {swap(nums[start], nums[end]);start++;end--;}
}// 轮转数组的主要函数
class Solution {
public:void rotate(vector<int>& nums, int k) {int n = nums.size();if (n == 0 || k % n == 0) return; // 如果数组为空或k是数组长度的倍数,不需要旋转// 反转整个数组reverse(nums, 0, n - 1);// 反转前 k 个元素reverse(nums, 0, k - 1);// 反转剩余的 n - k 个元素reverse(nums, k, n - 1);}
};// 主函数,用于演示
int main() {Solution solution;vector<int> nums = {1, 2, 3, 4, 5, 6, 7};int k = 3;cout << "Original array: ";for (int num : nums) {cout << num << " ";}cout << endl;solution.rotate(nums, k);cout << "Rotated array: ";for (int num : nums) {cout << num << " ";}cout << endl;return 0;
}
  • 输出结果:

Original array: {1, 2, 3, 4, 5, 6, 7}
Rotated array: {5, 6, 7, 1, 2, 3, 4}

其他题解

1. 额外空间法

这种方法使用额外的数组来存储元素,然后复制回原数组。这种方法简单,但不是原地操作。

void rotate(vector<int>& nums, int k) {vector<int> temp(nums.size());for (int i = 0; i < nums.size(); ++i) {temp[(i + k) % nums.size()] = nums[i];}nums = temp;
}

2. 反转法(原地操作)

这种方法通过三次反转来实现原地轮转,不需要额外的空间。

步骤:
  • 反转整个数组。
  • 反转前 k 个元素。
  • 反转 k 个元素后的剩余部分。
void rotate(vector<int>& nums, int k) {int n = nums.size();k %= n; // 避免 k 大于数组长度reverse(nums, 0, n - 1); // 反转整个数组reverse(nums, 0, k - 1); // 反转前 k 个元素reverse(nums, k, n - 1); // 反转剩余元素
}void reverse(vector<int>& nums, int start, int end) {while (start < end) {swap(nums[start], nums[end]);start++;end--;}
}

3. 循环交换法

这种方法通过循环交换元素来实现轮转,但可能需要多次循环。

void rotate(vector<int>& nums, int k) {int n = nums.size();k %= n;for (int i = 0; i < k; ++i) {swap(nums[i], nums[n - 1 - i]);}
}

4. 递归法

这种方法通过递归地将问题分解为更小的问题来解决。

void rotate(vector<int>& nums, int k) {k %= nums.size();if (k == 0) return;rotate(nums, k - 1);swap(nums[0], nums[nums.size() - k]);
}

5. 滑动窗口法

这种方法使用一个滑动窗口来逐步移动元素。

void rotate(vector<int>& nums, int k) {int n = nums.size();k %= n;for (int i = 0; i < k; ++i) {nums.push_back(nums[i]);}for (int i = 0; i < n; ++i) {nums[i] = nums[i + k];}nums.erase(nums.begin(), nums.begin() + k);
}

每种方法都有其优缺点,选择哪种方法取决于具体问题的要求和个人偏好。原地操作通常更优,因为它不需要额外的存储空间。

这篇关于LeetCode 算法:轮转数组c++的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1047263

相关文章

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个