题目地址:https://leetcode-cn.com/problems/er-wei-shu-zu-zhong-de-cha-zhao-lcof/

题目描述

在一个 n * m 的二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。

示例:

  1. 现有矩阵 matrix 如下:
  2. [
  3. [1, 4, 7, 11, 15],
  4. [2, 5, 8, 12, 19],
  5. [3, 6, 9, 16, 22],
  6. [10, 13, 14, 17, 24],
  7. [18, 21, 23, 26, 30]
  8. ]
  9. 给定 target = 5,返回 true
  10. 给定 target = 20,返回 false

限制:

0 <= n <= 1000

0 <= m <= 1000

注意:本题与主站 240 题相同:https://leetcode-cn.com/problems/search-a-2d-matrix-ii/

解题方法

注意每行都是排好序的,而且每列从上到下也都是排好序的。

  • 如果是从「左上角」开始遍历的话,当 target 比当前的元素大,我们就无法知道向「右」还是向「下」进行遍历。因此不能从「左上角」开始遍历。
  • 对于这个二维数组的查找有个很巧妙的方法,是从「左下角」或者「右上角」开始进行搜索。

以从右上角(0, col - 1)开始为例:

  • 如果 target 比当前值小,那么应该行减小;
  • 如果target比当前值大,列增加。
  • 直到寻找到 target。
  • 如果当搜索到二维数组之外了,说明找不到。

为什么可以这么做呢?在「宫水三叶」的题解中,提到了这种解释,让人茅塞顿开:

可以把数组抽象成一个 BST,比如以下图中的数字「7」作为 BST 的根节点,那么,如果要寻找的 target 比 7 小,就应该向 7 的左边寻找(类似于 BST 的左子树);如果 target 比 7 大,就应该向 7 的下边寻找(类似于右子树)。

java代码如下:

  1. public class Solution {
  2. public boolean Find(int target, int [][] array) {
  3. int row = 0;
  4. int col = array[0].length - 1;
  5. while(row < array.length && col >= 0){
  6. if(target == array[row][col]){
  7. return true;
  8. }else if(target > array[row][col]){
  9. row++;
  10. }else{
  11. col--;
  12. }
  13. }
  14. return false;
  15. }
  16. }

Python解法如下:

用python的时候,从左下角开始进行的搜索。

  1. # -*- coding:utf-8 -*-
  2. class Solution:
  3. # array 二维列表
  4. def Find(self, target, array):
  5. if len(array) == 0 or len(array[0]) == 0:
  6. return False
  7. rows, cols = len(array), len(array[0])
  8. i, j = rows - 1, 0
  9. while i >=0 and j < cols:
  10. if array[i][j] == target:
  11. return True
  12. elif array[i][j] > target:
  13. i -= 1
  14. else:
  15. j += 1
  16. return False

C++代码使用从右上角开始的搜索,代码如下:

  1. class Solution {
  2. public:
  3. bool findNumberIn2DArray(vector<vector<int>>& matrix, int target) {
  4. if (matrix.size() == 0 || matrix[0].size() == 0)
  5. return false;
  6. const int M = matrix.size();
  7. const int N = matrix[0].size();
  8. int i = 0;
  9. int j = N - 1;
  10. while (true) {
  11. if (i >= M || j < 0)
  12. break;
  13. if (matrix[i][j] == target) {
  14. return true;
  15. } else if (matrix[i][j] < target) {
  16. i ++;
  17. } else {
  18. j --;
  19. }
  20. }
  21. return false;
  22. }
  23. };

日期

2017 年 4 月 20 日
2018 年 3 月 9 日
2020 年 3 月 16 日 ——— 三年后再刷此题
2021 年 7 月 20 日 ——— 三年后再刷此题

【剑指Offer】04. 二维数组中的查找 解题报告(Java & Python & C++)的更多相关文章

  1. 剑指 Offer 04. 二维数组中的查找 (思维)

    剑指 Offer 04. 二维数组中的查找 题目链接 本题的解法是从矩阵的右上角开始寻找目标值. 根据矩阵的元素分布特性, 当目标值大于当前位置的值时将row行号++,因为此时目标值一定位于当前行的下 ...

  2. 剑指 Offer 04. 二维数组中的查找

    链接:https://leetcode-cn.com/problems/er-wei-shu-zu-zhong-de-cha-zhao-lcof/ 标签:数组.双指针.二分 题目 在一个 n * m ...

  3. 【LeetCode】剑指 Offer 04. 二维数组中的查找

    二维数组查找:线性查找法 有二维数组: [  [1,   4,  7, 11, 15],  [2,   5,  8, 12, 19],  [3,   6,  9, 16, 22],  [10, 13, ...

  4. 剑指offer——04二维数组中的查找

    题目: 数组中唯一只出现一次的数字.在一个数组中除一个数字只出现一次之外,其他数字都出现了三次.请找出那个只出现一次的数字. 题解: 如果一个数字出现三次,那么它的二进制表示的每一位(0或者1)也出现 ...

  5. 剑指offer:二维数组中的查找

    目录 题目 解题思路 具体代码 题目 题目链接 剑指offer:二维数组中的查找 题目描述 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺 ...

  6. 《剑指offer》 二维数组中的查找

    本题目是<剑指offer>中的题目 二维数组中的查找 题目: 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个 ...

  7. 【Java】 剑指offer(3) 二维数组中的查找

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集 题目 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上 ...

  8. [剑指Offer]5.二维数组中的查找

    题目 在一个二维数组中,每一行都依照从左到右递增的顺序排序,每一列都依照从上到下递增的顺序排序.请完毕一个函数,输入这种一个二维数组和一个整数.推断数组中是否含有该整数. 思路 [算法系列之三十三]杨 ...

  9. 《剑指Offer 1.二维数组中的查找》2019-03-25

    剑指Offer  第一题 题目描述 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数 ...

随机推荐

  1. 金蝶EAS——我的EAS报销流程怎么能让另一个人看到呢?即如何设置流程传阅功能?设置“代理报销”

    代理的话只能看到被代理人能看到的流程.设置"代理报销":应用--财务会计--费用管理--代理报销 选择报销人公司--"他人代理我报销"--选择报销人(zhaof ...

  2. Excel-统一小括号格式(中文小括号,英文小括号)

    1.统一小括号格式(中文小括号,英文小括号) 公式=ASC("(") #"(" 解释函数: ASC(A1)#对于双字节字符集(DBCS)语言,将全角英文字符(即 ...

  3. excel--CLEAN()函数,解决为什么看着相同的字符串但是len()长度不同

    CLEAN()函数能够有效解决去除字符串中隐藏的字符(这些字符是TRIM()去除不掉的)

  4. 63.不同路径II

    目录 63.不同路径Ⅱ 题目 题解 63.不同路径Ⅱ 题目 一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为"Start" ). 机器人每次只能向下或者向右移动 ...

  5. Spark(二十)【SparkSQL将CSV导入Kudu】

    目录 SparkSql 将CSV导入kudu pom 依赖 scala 代码 启动脚本 SparkSql 将CSV导入kudu pom 依赖 <properties> <spark. ...

  6. Android Handler 消息机制原理解析

    前言 做过 Android 开发的童鞋都知道,不能在非主线程修改 UI 控件,因为 Android 规定只能在主线程中访问 UI ,如果在子线程中访问 UI ,那么程序就会抛出异常 android.v ...

  7. Oracle中创建DB LINK

    当用户要跨本地数据库,访问另外一个数据库表中的数据时,本地数据库中必须创建了远程数据库的dblink,通过dblink本地数据库可以像访问本地数据库一样访问远程数据库表中的数据.下面讲介绍如何在本地数 ...

  8. Linux:cp -rp

    cp -rp[原文件或目录] [目标文件或目录] -r   复制目录 - p   保留文件属性 范例: cp -r /yy/k /yy/u /mm 复制目录u和目录k到目录mm中 cp -r /yy/ ...

  9. VFL

    VFL 1. 概念 VFL全称是Visual Format Language,翻译过来是"可视化格式语言" VFL是苹果公司为了简化Autolayout的编码而推出的抽象语言 2. ...

  10. Reactor之发射器(Flux、Mono)转换操作函数

    数据合并函数 由于业务需求有的时候需要将多个数据源进行合并,Reactor提供了concat方法和merge方法: concat public static <T> Flux<T&g ...