day5 题目:剑指 Offer 04. 二维数组中的查找、剑指 Offer 11. 旋转数组的最小数字、剑指 Offer 50. 第一个只出现一次的字符
知识点:数组、二分、哈希,难度为中等、简单、简单
学习计划链接:「剑指 Offer」 - 学习计划
题目 | 知识点 | 难度 |
---|---|---|
剑指 Offer 04. 二维数组中的查找 | 数组、二分 | 中等 |
剑指 Offer 11. 旋转数组的最小数字 | 数组、二分 | 简单 |
剑指 Offer 50. 第一个只出现一次的字符 | 字符串、哈希 | 简单 |
# 剑指 Offer 04. 二维数组中的查找
在一个 n * m 的二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个高效的函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。
示例:
现有矩阵 matrix 如下:
[
[1, 4, 7, 11, 15],
[2, 5, 8, 12, 19],
[3, 6, 9, 16, 22],
[10, 13, 14, 17, 24],
[18, 21, 23, 26, 30]
]
给定 target = 5
,返回 true
。
给定 target = 20
,返回 false
。
限制:
0 <= n <= 1000
0 <= m <= 1000
注意: 本题与主站 240 题相同:https://leetcode-cn.com/problems/search-a-2d-matrix-ii/
# 思路及代码
思路 1:二分法 + 一些优化
/** | |
* @param {number[][]} matrix | |
* @param {number} target | |
* @return {boolean} | |
*/ | |
var findNumberIn2DArray = function(matrix, target) { | |
if(matrix.length == 0) return false; | |
let [n, m] = [matrix.length, matrix[0].length]; | |
for(let k = 0; k < n; ++k) { | |
if(matrix[k][0] > target) | |
break; | |
if(matrix[k][m - 1] < target) | |
continue; | |
let [l, r] = [0, m-1] | |
while(l <= r) { | |
let mid = (l+r)>>1; | |
if(matrix[k][mid] === target) | |
return true; | |
if(matrix[k][mid] > target) r = mid - 1; | |
else l = mid + 1; | |
} | |
} | |
return false; | |
}; |
思路 2:站在数组右上角看这其实是一个二叉搜索树:二维数组中的查找 - 力扣
二叉搜索树中左边元素都比他小,右边元素都比他大,故这里若当前元素小于目标值则向右(实际是向下 r++)找,大于目标值则向左(实际是向左 c--)找。
var findNumberIn2DArray = function(matrix, target) { | |
if(matrix.length == 0) return false; | |
let [n, m] = [matrix.length, matrix[0].length]; | |
let [r, c] = [0, m - 1]; | |
while(r < n && c >= 0) { | |
if(matrix[r][c] == target) return true; | |
else if(matrix[r][c] > target) --c; | |
else ++r; | |
} | |
return false; | |
}; |
# 剑指 Offer 11. 旋转数组的最小数字
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。
给你一个可能存在 重复 元素值的数组 numbers
,它原来是一个升序排列的数组,并按上述情形进行了一次旋转。请返回旋转数组的最小元素。例如,数组 [3,4,5,1,2]
为 [1,2,3,4,5]
的一次旋转,该数组的最小值为 1。
注意,数组 [a[0], a[1], a[2], ..., a[n-1]]
旋转一次 的结果为数组 [a[n-1], a[0], a[1], a[2], ..., a[n-2]]
。
示例 1:
输入: numbers = [3,4,5,1,2]
输出: 1
示例 2:
输入: numbers = [2,2,2,0,1]
输出: 0
提示:
n == numbers.length
1 <= n <= 5000
-5000 <= numbers[i] <= 5000
numbers
原来是一个升序排序的数组,并进行了1
至n
次旋转
注意:本题与主站 154 题相同:https://leetcode-cn.com/problems/find-minimum-in-rotated-sorted-array-ii/
# 思路及代码
在之前做过的 33. 搜索旋转排序数组
有讲过:冲刺春招 - 精选笔面试 66 题大通关 day6
因为可能有重复的元素,故此题还需注意重复元素处的 --r
这题的题解可以看官方的很详细:旋转数组的最小数字
var minArray = function(numbers) { | |
if(numbers.length == 0) return 0; | |
let [l, r] = [0, numbers.length - 1]; | |
while(l < r) { | |
let mid = (l + r) >> 1; | |
if(numbers[mid] > numbers[r]) l = mid + 1; | |
else if(numbers[mid] < numbers[r]) r = mid; | |
else --r; | |
} | |
return numbers[l]; | |
}; |
# 剑指 Offer 50. 第一个只出现一次的字符
在字符串 s 中找出第一个只出现一次的字符。如果没有,返回一个单空格。 s 只包含小写字母。
示例 1:
输入:s = "abaccdeff"
输出:'b'
示例 2:
输入:s = ""
输出:' '
限制:
0 <= s 的长度 <= 50000
# 思路及代码
遍历一遍用哈希表存,出现过两次以上的都按两次算。
var firstUniqChar = function(s) { | |
let m = new Map(); | |
for (let i = 0; i < s.length; i++) | |
m.set(s[i], m.has(s[i]) ? 2 : 1); | |
for (let i = 0; i < s.length; i++) | |
if (m.get(s[i]) === 1) | |
return s[i]; | |
return ' '; | |
}; |