Rotate Array leetcode
Rotate an array of n elements to the right by k steps.
For example, with n = 7 and k = 3, the array [1,2,3,4,5,6,7] is rotated to [5,6,7,1,2,3,4].
Note:
Try to come up as many solutions as you can, there are at least 3 different ways to solve this problem.
设 数组长度为n,移动步数为k
思路整理:
1.是否能通过公式计算出移动后的位置呢?
解决:假设当前位置为 a0,则移动后的位置为 a1 = (a0 + k) % n;
2.如果我们让nums[a1] = nums[a0],那么nums[a1]该怎么处理?
解决:可以计算a1位置移动后的下一个位置a2 = (a1 + k) % n,然后让nums[a2] = nums[a1],依次类推,直到下一个位置an回到起点位置 也就是 a,这样循环下去,总会将所有元素替换掉吧。
3.我试着编写出一个循环程序,提交后发现程序存在严重漏洞,如果 n % k = 0 或者 k % n = 0,我们执行这个程序会发现不能移动所有元素,比如 n = 6,k = 3,执行的操作是a0->a3,a3->a0,我们会发现a1,a2...等元素并没有替换掉。那么该怎么处理这个问题呢?
解决:抓住n % k = 0 或者 k % n = 0这种情况做分析,我发现可以加入一层外层循环,还是上面例子,外层循环是 a0,a1,a2,这样就可以保证所有元素可以替换掉了。
4.外层循环的执行次数 c 该如何限制呢?
解决:我使用了归纳假设方法,首先假设n % k = 0,c = k;k % n = 0,c = n,其他情况的话 c = 1,这种假设经过程序测试,发现是错误的,假如n = 6, k = 4很明显还是不能遍历所有元素。然后我继续使用归纳方法,举出很多种n和k的组合,最后大胆的假设c是n和k的最大公约数,事实证明,这是正确的。也许这个结论证明并不严谨,但已经足够解决这个问题了。
写出程序如下:
void rotate(vector<int>& nums, int k) {
int n = nums.size();
if (n == || k == )
return;
// 求最大公约数
int a = n, c = k, b = a % c;
while (b != )
{
a = c;
c = b;
b = a % c;
}
// 依次替换
while (c--)
{
int oldNum = nums[c];
int curPos = (c + k) % n;
while (curPos != c)
{
int curNum = nums[curPos];
nums[curPos] = oldNum;
curPos = (curPos + k) % n;
oldNum = curNum;
}
nums[curPos] = oldNum;
}
}
这里要注意依次替换的时候,我们要使用一个临时变量oldNum来保存上一次被替换掉的元素。循环体内的赋值过程有些绕,要理清思绪。我在这里卡了一阵子,越想越头晕。。。
算法总结:
这个算法成功的将时间复杂度控制在O(n),而且空间复杂度是O(1),更可贵的是所有操作没有改变内存位置。可以说是非常理想了。
查看了下网上的其他方法,大体上有三种,
一种是将所有元素从后往前依次移动一个位置,然后外层循环k次。这种方法时间复杂度为O(n*k),不消说,没有多少效率。
代表算法(C语言实现):
void rotate(int nums[], int n, int k) {
int temp;
for (int step = ; step < k; step++) {
temp = nums[n-];
for (int i = n-; i > ; --i)
{
nums[i] = nums[i-];
}
nums[] = temp;
}
}
一种是反转数组的方法,先把整个数组reverse,然后把前面的reverse回来,再把后面的reverse回来,代表算法:
void rotate(int nums[], int n, int k) {
k = k % n;
reverse(nums, nums + n);
reverse(nums, nums + k);
reverse(nums + k, nums + n);
}
这个算法比较高明,时间复杂度可以控制在O(n),空间复杂度也是O(1),非常理想,但要看你的reverse实现是否高效
还有一种方法个人认为比较投机取巧,它使用C语言的内存操作函数,改变了内存地址,有一定风险,但算法确实高效
void rotate(int nums[], int n, int k) {
k = k % n;
if (k == ) return;
int *temp = new int[n];
memcpy(temp, nums+(n-k), sizeof(int)*k);
memcpy(temp+k, nums, sizeof(int)*(n-k));
memcpy(nums, temp, sizeof(int)*n);
delete[] temp;
}
算法——让人类思维更加理性
Rotate Array leetcode的更多相关文章
- 189. Rotate Array - LeetCode
Question 189. Rotate Array Solution 题目大意:数组中最后一个元素移到第一个,称动k次 思路:用笨方法,再复制一个数组 Java实现: public void rot ...
- LeetCode: Reverse Words in a String && Rotate Array
Title: Given an input string, reverse the string word by word. For example,Given s = "the sky i ...
- C++ STL@ list 应用 (leetcode: Rotate Array)
STL中的list就是一双向链表,可高效地进行插入删除元素. List 是 C++标准程式库 中的一个 类 ,可以简单视之为双向 连结串行 ,以线性列的方式管理物件集合.list 的特色是在集合的任何 ...
- 2016.5.16——leetcode:Rotate Array,Factorial Trailing Zeroe
Rotate Array 本题目收获: 题目: Rotate an array of n elements to the right by k steps. For example, with n = ...
- LeetCode 189. 旋转数组(Rotate Array)
189. 旋转数组 LeetCode189. Rotate Array 题目描述 给定一个数组,将数组中的元素向右移动 k 个位置,其中 k 是非负数. 示例 1: 输入: [1,2,3,4,5,6, ...
- 【LeetCode】Rotate Array
Rotate Array Rotate an array of n elements to the right by k steps. For example, with n = 7 and k = ...
- LeetCode Rotate Array 翻转数组
题意:给定一个数组,将该数组的后k位移动到前n-k位之前.(本题在编程珠玑中第二章有讲) 思路: 方法一:将后K位用vector容器装起来,再移动前n-k位到后面,再将容器内k位插到前面. class ...
- <LeetCode OJ> 189. Rotate Array
189. Rotate Array Total Accepted: 55073 Total Submissions: 278176 Difficulty: Easy Rotate an array o ...
- Python3解leetcode Rotate Array
问题描述: Given an array, rotate the array to the right by k steps, where k is non-negative. Example 1: ...
随机推荐
- java调用oracle存储过程,返回结果集
package com.srie.db.pro; import java.sql.CallableStatement; import java.sql.Connection; import java. ...
- Linux笔记(三) - 文件搜素
(1)文件搜索:find-name 根据文件名, *匹配任意字符 ,?单个字符-iname 根据文件名, 不区分大小写-size 根据文件大小查找 (+ 大于 -小于)(-a并且 -o或者)-us ...
- apache 运行php环境之困扰,无法加载多个不同的.html文件
又是一个项目,为多个纯静态html页面h5游戏页,原本是一个简单得不能的项目,但是却多生了事端. 我按照apache的惯例,将文件上传到服务器的DocumentRoot目录,进行测试了. 刚开始使用目 ...
- 不要怂,就是GAN (生成式对抗网络) (四):训练和测试 GAN
在 /home/your_name/TensorFlow/DCGAN/ 下新建文件 train.py,同时新建文件夹 logs 和文件夹 samples,前者用来保存训练过程中的日志和模型,后者用来保 ...
- 编写JQuery插件-2
继续上一节的代码 (function(){ /* code */ })() 来我们介绍一下吧,首先定义一个匿名函数 fnction(){/* 这里放置代码 */} 然后用括号括起来, (fnction ...
- 关于j2ee工程发布到was上后,部分更新,例修改web.xml配置文件不起作用的原因解析【转】
在WAS中,应用的配置是从config/cells....目录下读取:而资源从/installedApps目录下读取 故当配置文件(例web.xml)发生改变时,只更新应用程序资源文件/install ...
- 时间处理之strtotime
strtotime (PHP 4, PHP 5, PHP 7)strtotime - 将任何英文文本的日期时间描述解析为 Unix 时间戳说明 int strtotime ( string $time ...
- Android jni 编程1(对基本类型字符串的操作)
最近一直在学安卓的jni,主要参考的是黑马程序员的视频教程,讲的确实不错. 那就自己总结一下吧,算是对学习的复习. 这篇博客也主要参考了这位博主:http://www.cnblogs.com/acti ...
- matlab for循环应用(阶乘及the day of year)
一.N的阶乘 %脚本文件:test.m %N的阶乘 使用举例 % 定义变量 % ii ---循环变量,也就是循环次数 % N ---N的阶乘 % N_factorial --计算N的阶乘 clc;cl ...
- 如何一步一步用DDD设计一个电商网站(十四)—— 回顾与总结
本系列所有文章 如何一步一步用DDD设计一个电商网站(一)—— 先理解核心概念 如何一步一步用DDD设计一个电商网站(二)—— 项目架构 如何一步一步用DDD设计一个电商网站(三)—— 初涉核心域 如 ...