LeetCodeCapmusR2Day01-数组
LeetCodeCapmusR2Day01-数组
第二轮刷题,期望是包含指定任务题目外添加一些额外题目
704. 二分查找
https://leetcode.cn/problems/binary-search/
给定一个 n
个元素有序的(升序)整型数组 nums
和一个目标值 target
,写一个函数搜索 nums
中的 target
,如果 target
存在返回下标,否则返回 -1
。
你必须编写一个具有 O(log n)
时间复杂度的算法。
示例 1:
1 | 输入: nums = [-1,0,3,5,9,12], target = 9 |
示例 2:
1 | 输入: nums = [-1,0,3,5,9,12], target = 2 |
提示:
- 你可以假设
nums
中的所有元素是不重复的。 n
将在[1, 10000]
之间。nums
的每个元素都将在[-9999, 9999]
之间。
代码
- 时间复杂度O(LogN)
- 空间复杂度O(1)
1 | class Solution: |
27. 移除元素
https://leetcode.cn/problems/remove-element/
给你一个数组 nums
和一个值 val
,你需要 原地 移除所有数值等于 val
的元素。元素的顺序可能发生改变。然后返回 nums
中与 val
不同的元素的数量。
假设 nums
中不等于 val
的元素数量为 k
,要通过此题,您需要执行以下操作:
- 更改
nums
数组,使nums
的前k
个元素包含不等于val
的元素。nums
的其余元素和nums
的大小并不重要。 - 返回
k
。
用户评测:
评测机将使用以下代码测试您的解决方案:
1 | int[] nums = [...]; // 输入数组 |
如果所有的断言都通过,你的解决方案将会 通过。
示例 1:
1 | 输入:nums = [3,2,2,3], val = 3 |
示例 2:
1 | 输入:nums = [0,1,2,2,3,0,4,2], val = 2 |
提示:
0 <= nums.length <= 100
0 <= nums[i] <= 50
0 <= val <= 100
代码
1 | class Solution: |
977. 有序数组的平方
https://leetcode.cn/problems/squares-of-a-sorted-array/
给你一个按 非递减顺序 排序的整数数组 nums
,返回 每个数字的平方 组成的新数组,要求也按 非递减顺序 排序。
示例 1:
1 | 输入:nums = [-4,-1,0,3,10] |
示例 2:
1 | 输入:nums = [-7,-3,2,3,11] |
提示:
1 <= nums.length <= 104
-104 <= nums[i] <= 104
nums
已按 非递减顺序 排序
进阶:
- 请你设计时间复杂度为
O(n)
的算法解决本问题
代码
- 时间复杂度O(N)
- 空间复杂度O(N)
1 | class Solution: |
922. 按奇偶排序数组 II
https://leetcode.cn/problems/sort-array-by-parity-ii/
给定一个非负整数数组 nums
, nums
中一半整数是 奇数 ,一半整数是 偶数 。
对数组进行排序,以便当 nums[i]
为奇数时,i
也是 奇数 ;当 nums[i]
为偶数时, i
也是 偶数 。
你可以返回 任何满足上述条件的数组作为答案 。
示例 1:
1 | 输入:nums = [4,2,5,7] |
示例 2:
1 | 输入:nums = [2,3] |
提示:
2 <= nums.length <= 2 * 104
nums.length
是偶数nums
中一半是偶数0 <= nums[i] <= 1000
**进阶:**可以不使用额外空间解决问题吗
代码
- 时间复杂度O(N)
- 这里时间复杂度并不是O(n^2),因为偶数位和奇数位都只操作一次,不是n/2 * n/2的关系,而是n/2 + n/2的关系!
- 空间复杂度O(1)
1 | class Solution: |
35. 搜索插入位置
https://leetcode.cn/problems/search-insert-position/
给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。
请必须使用时间复杂度为 O(log n)
的算法。
示例 1:
1 | 输入: nums = [1,3,5,6], target = 5 |
示例 2:
1 | 输入: nums = [1,3,5,6], target = 2 |
示例 3:
1 | 输入: nums = [1,3,5,6], target = 7 |
提示:
1 <= nums.length <= 104
-104 <= nums[i] <= 104
nums
为 无重复元素 的 升序 排列数组-104 <= target <= 104
代码
- 时间复杂度O(logN)
- 空间复杂度O(1)
1 | class Solution: |
41. 缺失的第一个正数
https://leetcode.cn/problems/first-missing-positive/
给你一个未排序的整数数组 nums
,请你找出其中没有出现的最小的正整数。
请你实现时间复杂度为 O(n)
并且只使用常数级别额外空间的解决方案。
示例 1:
1 | 输入:nums = [1,2,0] |
示例 2:
1 | 输入:nums = [3,4,-1,1] |
示例 3:
1 | 输入:nums = [7,8,9,11,12] |
提示:
1 <= nums.length <= 105
-231 <= nums[i] <= 231 - 1
一般思路
可以从1开始依次枚举正整数,并且遍历数组;但这样的时间复杂度为O(N^2),空间复杂度为O(1);
当然,可以使用hashtable,将时间复杂度降到O(N),但空间复杂度会为O(N);因为hashtable的查找时间是O(1)。我们为什么要使用哈希表?这是因为哈希表是一个可以支持快速查找的数据结构:给定一个元素,我们可以在 *O*(1) 的时间查找该元素是否在哈希表中
进阶思路
可以先将数组排序,然后对数据进行遍历,所有0以及负数可以不管,再按正数顺序查找数组中可以达到的不存在的最小正数;
但这个对数组排序仍然会占用更多空间
进阶思路代码
- 时间复杂度O(N)
- 空间复杂度O(N) # 对数组排序需要占用空间
1 | class Solution: |
动脑筋思路
那是否有什么方法,允许对nums数组进行修改,并且保持O(N)的时间复杂度呢?
本题目如果是有序的数组,那就很好解决了,但问题就出在现在是无序的
;
下面有个不错的思路:
题目要求一个长度为N的数组,所以没有出现的最小正整数的范围一定是[1, N + 1](都是闭区间),我们对数组进行遍历,对于遍历到的数x,如果它在[1, N]范围内,则将数组中的第x - 1位置(因为下标从0开始)打上标记
,如果在遍历结果后,所有位置都被打了标记,则说明最小正整数是N+1;否则,最小正整数就是没有标记的下标+1
比如n = 6
3 | 4 | -1 | 1 | 9 | -5 | 1.将负数变为n+1 |
---|---|---|---|---|---|---|
3 | 4 | 7 | 1 | 9 | 7 | 2.将小于n=6的元素,它对应位置变成负数 |
-3 | 4 | -7 | -1 | 9 | 7 | 3.返回第一个大于0的下标+1 |
注意,这里的第二步:将小于n=6的元素,它对应位置变成负数
比如第一个元素3,则将nums[3 - 1] = -abs(nums[3 - 1]),那就是将下标为2的原元素7变成1了-7
比如元素1,则将nums[1 - 1] = -abs(nums[1 - 1]),将下标为0的原元素3变成了-3
也就是说,如果某个正数在数组里出现了,那把这个正数当作下标,将下标所在的数字打上标记(这里是将其值设置为负数)
比如下图中,缺少
正数2
,所以下标nums[2 - 1]的数字不会被打上标记!最终我们只要找哪个数字没有标记!它的下标就是我们要找的最小正数
下面这个图看得更清楚