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

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 4

int 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 所在的列,
  • 这样我们就无法缩小查找范围了

相关推荐

  1. offer面试题3 数组查找

    2024-02-14 00:46:02       32 阅读
  2. 力扣【offer数组查找

    2024-02-14 00:46:02       12 阅读
  3. 矩阵二分查找算法实现

    2024-02-14 00:46:02       37 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-02-14 00:46:02       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-02-14 00:46:02       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-02-14 00:46:02       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-02-14 00:46:02       20 阅读

热门阅读

  1. 力扣:376. 摆动序列

    2024-02-14 00:46:02       30 阅读
  2. 交易中的胜率和盈亏比估算

    2024-02-14 00:46:02       54 阅读
  3. 数据结构-树

    2024-02-14 00:46:02       28 阅读
  4. day2-理解 linux 云计算

    2024-02-14 00:46:02       32 阅读
  5. C#中 Combine 静态方法

    2024-02-14 00:46:02       29 阅读
  6. STM32 与 ARM 谁比较强大?

    2024-02-14 00:46:02       28 阅读
  7. ndk-r20b 编译 boost 1.74。

    2024-02-14 00:46:02       35 阅读
  8. 遗传算法实现

    2024-02-14 00:46:02       27 阅读
  9. 安卓termux mosh配置nvim远程开发

    2024-02-14 00:46:02       36 阅读
  10. A股上市以来涨幅排行榜

    2024-02-14 00:46:02       34 阅读