day5题目:剑指 Offer 04. 二维数组中的查找 、剑指 Offer 11. 旋转数组的最小数字 、剑指 Offer 50. 第一个只出现一次的字符
知识点:数组、二分、哈希,难度为中等、简单、简单
学习计划链接:「剑指 Offer」 - 学习计划
在一个 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;
};
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。
给你一个可能存在 重复 元素值的数组 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
提示:
-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];
};
在字符串 s 中找出第一个只出现一次的字符。如果没有,返回一个单空格。 s 只包含小写字母。
示例 1:
复制 输入:s = "abaccdeff"
输出:'b'
示例 2:
限制:
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 ' ';
};