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 <= 1000 <= nums[i] <= 500 <= 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] <= 104nums已按 非递减顺序 排序
进阶:
- 请你设计时间复杂度为 
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 * 104nums.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] <= 104nums为 无重复元素 的 升序 排列数组-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]的数字不会被打上标记!最终我们只要找哪个数字没有标记!它的下标就是我们要找的最小正数
下面这个图看得更清楚









