初级算法 | 数组——旋(轮)转数组

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

 

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

解题:

(1)引入新数组

遍历赋值,将原数组下标为 i 的元素赋值给新数组下标为 (i+k)%numsSize 的元素

void rotate(int* nums, int numsSize, int k){
    int newArr[numsSize];
    for (int i = 0; i < numsSize; ++i) {
        newArr[(i + k) % numsSize] = nums[i];
    }
    for (int i = 0; i < numsSize; ++i) {
        nums[i] = newArr[i];
    }

}

(2)矩阵翻转

翻转整个函数,再以nums[k]为分界,分别翻转拆分的两部分数组

void swap(int* a, int* b) {
    int t = *a;
    *a = *b;
    *b = t;
}
void reverse(int* nums, int start, int end){
    while(start < end){
        swap(&nums[start],&nums[end]);
        start++;
        end--;
    }
}
void rotate(int* nums, int numsSize, int k){
    k %= numsSize;
    reverse(nums, 0, numsSize - 1);
    reverse(nums, 0, k - 1);
    reverse(nums, k, numsSize - 1);
}

(3)环状替代

        从位置 0 开始,令 temp=nums[0]。位置 0 的元素放至 (0+k) % n 的位置,令 x=(0+k) % n,交换temp 和 nums[(0+k) % n],完成 x 的更新。然后,继续交换 temp 和 nums[(x+k)%n],完成下一个位置的更新。不断重复,直至回到初始位置 0。

        需要像上述遍历的次数为:gcd(n,k),即n和k的最大公约数

int gcd(int a, int b) {
    return b ? gcd(b, a % b) : a;
}

void swap(int* a, int* b) {
    int t = *a;
    *a = *b, *b = t;
}

void rotate(int* nums, int numsSize, int k) {
    k = k % numsSize;
    int count = gcd(k, numsSize);
    for (int start = 0; start < count; ++start) {
        int current = start;
        int prev = nums[start];
        do {
            int next = (current + k) % numsSize;
            swap(&nums[next], &prev);
            current = next;
        } while (start != current);
    }
}

class Solution {
    public void rotate(int[] nums, int k) {
        int n = nums.length;
        k = k % n;
        int count = gcd(k, n);
        for (int start = 0; start < count; ++start) {
            int current = start;
            int prev = nums[start];
            do {
                int next = (current + k) % n;
                int temp = nums[next];
                nums[next] = prev;
                prev = temp;
                current = next;
            } while (start != current);
        }
    }

    public int gcd(int x, int y) {
        return y > 0 ? gcd(y, x % y) : x;
    }
}

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值