由于最终回到了起点,故该过程恰好走了整数数量的圈,不妨设为 a 圈;再设该过程总共遍历了 b 个元素。因此,我们有 an=bk,即 an 一定为 n,k 的公倍数。又因为我们在第一次回到起点时就结束,因此 a 要尽可能小,故 an 就是 n,k 的最小公倍数 lcm(n,k),因此 b 就为 lcm(n,k)/k。
classSolution { public: voidrotate(vector<int>& nums, int k){ int n = nums.size(); vector<int> newArr(n); for (int i = 0; i < n; ++i) { newArr[(i + k) % n] = nums[i]; } nums.assign(newArr.begin(), newArr.end()); } };
2、数组翻转
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
classSolution { public: voidreverse(vector<int>& nums, int start, int end){ while (start < end) { swap(nums[start], nums[end]); start += 1; end -= 1; } }
voidrotate(vector<int>& nums, int k){ k %= nums.size(); reverse(nums, 0, nums.size() - 1); reverse(nums, 0, k - 1); reverse(nums, k, nums.size() - 1); } };
3、环状替换
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
classSolution { public: voidrotate(vector<int>& nums, int k){ int n = nums.size(); 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; swap(nums[next], prev); current = next; } while (start != current); } } };
📊 复杂度分析
1、使用额外数组
时间复杂度:O(n),其中 n 为数组的长度。
空间复杂度:O(n)。
2、数组翻转
时间复杂度:O(n),其中 n 为数组的长度。每个元素被翻转两次,一共 n 个元素,因此总时间复杂度为 O(2n)=O(n)。