剑指offer——二维数组中的查找(杨氏矩阵)

2024-02-10 04:36

本文主要是介绍剑指offer——二维数组中的查找(杨氏矩阵),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 1. 题目描述
  • 2. 常见错误思路
  • 3. 分析
    • 3.1 特例分析
    • 3.2 规律总结
  • 4. 完整代码

1. 题目描述

  • 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。
  • 例如下面的二维数组就是每行、每列都递增排序。如果在这个数组中查找数字 7,则返回 true ;如果查找数字 5,由于数组不含该数字,则返回 false。

在这里插入图片描述

2. 常见错误思路

  • 分析这个问题的时候,很多应聘者都会把二维数组画成矩形,从数组中选取一个数字,分3种情况来分析查找的过程。
  • 当数组中选取的数字刚好和要查找的数字相等时,就结束查找过程。如果选取的数字小于要查找的数字,那么根据数组排序的规则,要查找的数字应该在当前选取的位置的右边或者下边(如图2.1(a)所示)。
  • 同样,如果选取的数字大于要查找的数字,那么要查找的数字应该在当前选取的位置的上边或者左边(如图2.1(b)所示).

在这里插入图片描述

  • 注:在数组中间选择一个数(深色方格),根据它的大小判断要查找的数字可能的区域(阴影部分)
  • 在上面的分析中,由于要查找的数字相对于当前选取的位置有可能在两个区域中出现,而且这两个区域还有重叠,这问题看起来就复杂了,于是很多人就卡在这里束手无策了。

3. 分析

3.1 特例分析

  • 当我们需要解决一个复杂的问题时,一个很有效的办法就是从一个具体的问题入手,通过分析简单具体的例子,试图寻找普遍的规律。
  • 针对这个问题,我们不妨也从一个具体的例子入手。下面我们以在题目中给出的数组中查找数字7为例来一步步分析查找的过程。
  • 前面我们之所以遇到难题,是因为我们在二维数组的中间选取一个数字来和要查找的数字做比较,这样导致下一次要查找的是两个相互重叠的区域。如果我们从数组的一个角上选取数字来和要查找的数字做比较,情况会不会变简单呢?
  • 首先我们选取数组右上角的数字9。由于9大于7,并且9还是第4列的第一个(也是最小的)数字,因此7不可能出现在数字9所在的列。
  • 于是我们把这一列从需要考虑的区域内剔除,之后只需要分析剩下的3列(如图22(a)所示)。在剩下的矩阵中,位于右上角的数字是8。
  • 同样8大于7,因此8所在的列我们也可以剔除。接下来我们只要分析剩下的两列即可(如图2.2(b)所示).
  • 在由剩余的两列组成的数组中,数字2位于数组的右上角。
  • 2小于7,那么要查找的7可能在2的右边,也有可能在2的下边。在前面的步骤中,我们已经发现2右边的列都已经被剔除了,也就是说7不可能出现在2的右边,
  • 因此7只有可能出现在2的下边,于是我们把数字2所在的行也别除,只分析剩下的三行两列数字(如图2.2(c)所示)。在剩下的数字中,数字4位于右上角,和前面一样,我们把数字4所在的行也别除,最后剩下两列数字(如图2.2(d)所示)。
  • 在剩下的两行两列 4 个数字中,位于右上角的刚好就是我们要查找的数字 7 ,于是查找过程就可以结束了

在这里插入图片描述

  • 注:矩阵中加阴影背景的区域是下一步查找的范围。

3.2 规律总结

  • 总结上述查找的过程,我们发现如下规律:首先选取数组中右上角的数字。
  • 如果该数字等于要查找的数字,查找过程结束;
  • 如果该数字大于要查找的数字,剔除这个数字所在的列;
  • 如果该数字小于要查找的数字,剔除这个数字所在的行。
  • 也就是说如果查找的数字不在数组的右上角,则每一次都在数组的查找范围中剔除一行或者一列,
  • 这样每一步都可以缩小查找的范围,直到找到要查找的数字,或者查找范围为空

4. 完整代码

  • 把整个查找过程分析清楚之后,我们再写代码就不是一件很难得事了,下面就是上述思路对应的参考代码
#include <stdio.h>
#define ROW 4
#define COL 4int findnum(int a[ROW][COL], int x, int y, int f)
{int i = 0, j = y - 1; //从右上角开始遍历while (j >= 0 && i < x){if (a[i][j] < f) //比我大就向下{i++;}else if (a[i][j] > f) //比我小就向左{j--;}else{return 1;}}return 0;
}int main()
{int a[ROW][COL] = { {1, 2, 8, 9 },{2, 4, 9, 12 },{4, 7, 10, 13 },{ 6, 8, 11, 15 } }; //一个示例int k = 0;scanf("%d", &k);//输入要查找的数字if (findnum(a, ROW, COL, k)){printf("It has been found!\n");}else{printf("It hasn't been found!\n");}return 0;
}
  • 运行结果如下:
  1. 当输入数组中有的数字时:

在这里插入图片描述

    1. 当输入数组没有的数字时:

在这里插入图片描述

  • 在前面的分析中,我们每一次都是选取数组查找范围内的右上角数字。
    同样,我们也可以选取左下角的数字。感兴趣的同学不妨自己分析一下每次都选取左下角的查找过程
  • 但我们不能选择左上角或者右下角。
  • 以左上角为例,最初数字 1 位于初始数组的左上角,由于 1 小于 7 ,那么应该位于 1 的右边或者下边。
  • 此时我们既不能从查找范围内剔除 1 所在的行,也不能剔除 1 所在的列,
  • 这样我们就无法缩小查找范围了

这篇关于剑指offer——二维数组中的查找(杨氏矩阵)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/696139

相关文章

MySQL JSON 查询中的对象与数组技巧及查询示例

《MySQLJSON查询中的对象与数组技巧及查询示例》MySQL中JSON对象和JSON数组查询的详细介绍及带有WHERE条件的查询示例,本文给大家介绍的非常详细,mysqljson查询示例相关知... 目录jsON 对象查询1. JSON_CONTAINS2. JSON_EXTRACT3. JSON_TA

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

C/C++中OpenCV 矩阵运算的实现

《C/C++中OpenCV矩阵运算的实现》本文主要介绍了C/C++中OpenCV矩阵运算的实现,包括基本算术运算(标量与矩阵)、矩阵乘法、转置、逆矩阵、行列式、迹、范数等操作,感兴趣的可以了解一下... 目录矩阵的创建与初始化创建矩阵访问矩阵元素基本的算术运算 ➕➖✖️➗矩阵与标量运算矩阵与矩阵运算 (逐元

JAVA数组中五种常见排序方法整理汇总

《JAVA数组中五种常见排序方法整理汇总》本文给大家分享五种常用的Java数组排序方法整理,每种方法结合示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录前言:法一:Arrays.sort()法二:冒泡排序法三:选择排序法四:反转排序法五:直接插入排序前言:几种常用的Java数组排序

C#实现查找并删除PDF中的空白页面

《C#实现查找并删除PDF中的空白页面》PDF文件中的空白页并不少见,因为它们有可能是作者有意留下的,也有可能是在处理文档时不小心添加的,下面我们来看看如何使用Spire.PDFfor.NET通过C#... 目录安装 Spire.PDF for .NETC# 查找并删除 PDF 文档中的空白页C# 添加与删

Java数组初始化的五种方式

《Java数组初始化的五种方式》数组是Java中最基础且常用的数据结构之一,其初始化方式多样且各具特点,本文详细讲解Java数组初始化的五种方式,分析其适用场景、优劣势对比及注意事项,帮助避免常见陷阱... 目录1. 静态初始化:简洁但固定代码示例核心特点适用场景注意事项2. 动态初始化:灵活但需手动管理代

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

Windows系统下如何查找JDK的安装路径

《Windows系统下如何查找JDK的安装路径》:本文主要介绍Windows系统下如何查找JDK的安装路径,文中介绍了三种方法,分别是通过命令行检查、使用verbose选项查找jre目录、以及查看... 目录一、确认是否安装了JDK二、查找路径三、另外一种方式如果很久之前安装了JDK,或者在别人的电脑上,想