当前位置:   article > 正文

LeetCode:189. 轮转数组(Java)_leetcode189java

leetcode189java

方法1: 跳序 轮转法——带标记,时间复杂度和空间复杂度均为O(n)

因为有些特殊情况,会陷入循环,比如这个例子:

我不知道怎么处理这种情况,所以直接搞个标记数组falgs[]来看看这个位置的数字是否被处理过。方法二来改进这个陷入循环的问题

  1. class Solution {
  2. public void rotate(int[] nums, int k) {
  3. // 1.跳序 轮转法——带标记,时间复杂度和空间复杂度均为O(n)
  4. int len=nums.length;
  5. int[] flags=new int[len];
  6. Arrays.fill(flags,0);
  7. int next;
  8. int curr=0;
  9. int tmp=nums[curr]; //临时保存上一个
  10. int tmp1; //临时保存当前的
  11. int n=len;
  12. while(n!=0 && k!=0){
  13. // if(n!=len && len%k==0 &&n%(len/k)==0){
  14. // curr++;
  15. // tmp=nums[curr];
  16. // }
  17. next=(curr+k)%len;
  18. while(flags[next]==1){
  19. next=(next+1)%len;
  20. if(flags[next]==1)
  21. continue;
  22. else {
  23. curr=next;
  24. tmp = nums[curr];
  25. next = (curr + k) % len;
  26. }
  27. }
  28. tmp1=nums[next];
  29. nums[next]=tmp;
  30. tmp=tmp1;
  31. curr=next;
  32. flags[curr]=1;
  33. n--;
  34. }
  35. }
  36. }

方法2:类1-跳序轮转法——不借助flag数组 

这里稍微对pos==i解释一下,因为pos如果一次陷入循环,意味着pos会在不同的轮转中循环,即跳出一个轮回会进入下一个轮换,在每个轮回中都会陷入循环。pos是从0开始,那当pos跳转一轮回到起点0以后,就派i(可以看做pos的初始)对线,发现跟初始0一样,就开始对pos+1处理。

(我们把这样叫一次轮转,从pos=0出发,回到pos=0)

  1. class Solution {
  2. public void rotate(int[] nums, int k) {
  3. // 5.类1-跳转轮转法:他人解法,不借助flag数组
  4. int len = nums.length,n = len;
  5. int i = 0,pos = 0, pre = nums[pos],temp = nums[pos];
  6. if(k%n == 0) return;
  7. while (n-- != 0) {
  8. pos = (pos + k) % len;
  9. temp = nums[pos];
  10. nums[pos] = pre;
  11. pre = temp;
  12. if (pos == i) { // 避免陷入循环
  13. pos = ++i;
  14. pre = nums[pos];
  15. temp = nums[pos];
  16. }
  17. }
  18. }
  19. }

方法3:借助另外一个数组保存每次移动的数——以空间换时间

  1. class Solution {
  2. public void rotate(int[] nums, int k) {
  3. // 双数组:顺序 轮转法——借助另一个数组。
  4. int len=nums.length;
  5. int[] res=Arrays.copyOf(nums,len);
  6. int next;
  7. for(int i=0;i<len;i++){
  8. next=(i+k)%len;
  9. nums[next]=res[i];
  10. }
  11. }
  12. }

方法4:翻转数组——有点巧妙的

  1. class Solution {
  2. public void rotate(int[] nums, int k) {
  3. // 4.三次翻转数组,找出k%n的那个位置点——耗时最少
  4. int len=nums.length;
  5. int pos=k%(len);
  6. reverse(nums,0,len);
  7. reverse(nums,0,pos);
  8. reverse(nums,pos,len);
  9. }
  10. public void reverse(int[] nums,int start,int end){
  11. int tmp;
  12. int i=0;
  13. while(i<(end-start)/2){
  14. tmp=nums[end-1-i];
  15. nums[end-1-i]=nums[start+i];
  16. nums[start+i]=tmp;
  17. i++;
  18. }
  19. }
  20. }

方法5:每次向前移动一位,重复k次(超出时间限制)

  1. class Solution {
  2. public void rotate(int[] nums, int k) {
  3. // 3.单次轮转k次法O(n*k)——超出时间限制×
  4. int len=nums.length;
  5. int tmp;
  6. while((k--)!=0){
  7. tmp=nums[0];
  8. for(int i=1;i<len;i++){
  9. nums[0]=nums[i];
  10. nums[i]=tmp;
  11. tmp=nums[0];
  12. }
  13. nums[0]=tmp;
  14. }
  15. }
  16. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/你好赵伟/article/detail/856202
推荐阅读
相关标签
  

闽ICP备14008679号