旋转数组 题目链接:https://leetcode-cn.com/problems/rotate-array/
1 2 3 4 5 题目: 给定一个数组,将数组中的元素向右移动 k 个位置,其中 k 是非负数。 示例: [1,2,3,4,5,6,7] 和 k = 3 结果: [5,6,7,1,2,3,4] 说明: 尽可能想出更多的解决方案,至少有三种不同的方法可以解决这个问题。 要求使用空间复杂度为 O(1) 的 原地 算法。
解法一: 暴力解决 该解决方案虽然满足题意中的空间复杂度条件,但是时间复杂度为O(k*n)
1 2 3 4 5 6 7 8 9 10 11 public void rotate (int nums[], int k) { k = k % nums.length; for (int j = 0 ; j < k; j++) { int pre = nums[nums.length - 1 ]; for (int i = 0 ; i < nums.length; i++) { int curr = nums[i]; nums[i] = pre; pre = curr; } } }
时间复杂度 O (k*n),k 次n个元素的遍历
空间复杂度 O (1) ,无额外空间占用
像这种暴力解决方案一般不是题目的解决方案。而题目中有说明至少有三种不同方法可以解决这个问题,所以还应该有其他的解决思路,本文列举了如下几种另外的解决思路
解法二: 引入额外数组 该方式和前面暴力解题方式是最简单的解题思路,但是不满足题目条件,题目中有说明用空间复杂度为O(1) 的原地算法,而这种解题方案的空间复杂度为O(n),故不满足要求
1 2 3 4 5 6 7 8 9 10 public void rotate (int [] nums, int k) { int [] arr = new int [nums.length]; for (int i = 0 ; i <nums.length;i++){ arr[(i+k)%nums.length] = nums[i]; } for (int i = 0 ; i < nums.length; i++) { nums[i] = arr[i]; } }
时间复杂度:O(n)
空间复杂度:O(n)
解法三: 环状替换 环状替换主要考虑两种情况,一种是数组长度是奇数的情况,这种情况下,形成环的方式只有一种。另一种情况也就是长度是偶数的情况,会形成多个环的情况
具体可以参考:https://leetcode-cn.com/problems/rotate-array/solution/xuan-zhuan-shu-zu-yuan-di-huan-wei-xiang-xi-tu-jie/
如下截图来自该链接。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 public void rotate (int nums[], int k) { k %= nums.length; int count = 0 ; for (int start = 0 ; count < nums.length; start++) { int currIndex = start; int move = nums[currIndex]; do { int byReplaceIndex = (currIndex + k) % nums.length; int byReplace = nums[byReplaceIndex]; nums[byReplaceIndex] = move; move = byReplace; currIndex = byReplaceIndex; count++; }while (start != currIndex); } }
时间复杂度 O (n),空间复杂度 O (1)
解法四: 数组反转 数组反转这个思想是参考官网解题思路的,通过三次反转操作,很巧妙的完成该题目的解答。
1 2 3 4 5 6 7 8 9 10 11 12 13 public void rotate (int [] nums, int k) { k %= nums.length; reverse(nums, 0 , nums.length - 1 ); reverse(nums, 0 , k - 1 ); reverse(nums, k, nums.length - 1 ); }public static void reverse (int nums[],int start,int end) { while (start < end) { nums[start] = nums[start] + nums[end]; nums[end] = nums[start] - nums[end]; nums[start] = nums[start++] - nums [end--]; } }
时间复杂度 O (n),空间复杂度 O (1)
链表反转 题目地址:https://leetcode-cn.com/problems/reverse-linked-list/
1 2 3 4 5 6 7 题目: 反转一个单链表。 示例 输入: 1->2->3->4->5->NULL 输出: 5->4->3->2->1->NULL 进阶: 你可以迭代或递归地反转链表。你能否用两种方法解决这道题?
Node 节点类辅助类
1 2 3 4 5 6 7 class ListNode { int val; ListNode next; ListNode(int x) { val = x; } }
解法一:双指针迭代 思路:题目中的 1 ← 2 ← 3 ← 4 ← 5 ← null 转换后为 null ← 1 ← 2 ← 3 ← 4 ← 5,所以可以定义两个node节点,一个记录pre,另一个记录next,然后依次遍历赋值即可
1 2 3 4 5 6 7 8 9 10 11 12 public static ListNode reverseList (ListNode head) { ListNode current = head; ListNode next = null ; ListNode temp = null ; while (current != null ) { next = current.next; current.next = temp; temp = current; current = next; } return temp; }
时间复杂度:O(n),空间复杂度:O(1)
解法二:递归(理解了一会儿) 思路:参考代码注释
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 public ListNode reverseList (ListNode head) { if (head == null || head.next == null ) { return head; } ListNode result = reverseList(head.next); head.next.next = head; head.next = null ; return result ; }
时间复杂度和空间复杂度都为:O(n)