凡事都有可能,永远别说永远。——《放牛班的春天》

今天一题为再一个行列都有序的二维数组中寻找一个目标值,我们第一时间想到的可能是很暴力的解法,例如从头到尾进行遍历,这样能做出来,但是借用武忠祥老师的一句话:这样做你就慢了,没效率。

本文章会从复杂度的角度来分析一个算法,循序渐进的提升这个算法的优秀程度。最终优化为最优算法。

一.题目及示例

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

[
[1,2,8,9],
[2,4,9,12],
[4,7,10,13],
[6,8,11,15]
]
给定 target = 7,返回 true。
给定 target = 3,返回 false。

下面给出示例:

解法一(暴力解法):

拿到这道题目,第一种方法不难想到,就是对整个数组进行遍历,每一个元素都扫一遍。这种解法比较暴躁,很暴力,给它一个外号,叫"暴力解法"。

具体代码实现如下:

bool Find(int target, vector<vector<int> > array) {// 利用C++的auto进行遍历for(auto x:array) {for(auto y:x){// 如果存在,直接返回trueif(y==target){return true;}}}return false;

ps:这里的for循环不是普通的for循环,这是C++新定义的for循环,具体使用可以去看我下一篇博客。

我把他称之为升级版for循环。

复杂度分析:

  • 需要对二维数组的每个元素进行遍历,时间复杂度为O(n^2)

  • 存储一个二维数组,空间复杂度为O(1)

解法二(双指针法)

由于矩阵行列都是严格递增的,此矩阵为杨氏矩阵,故也称杨氏矩阵法。

  • 如果这个是一个没有任何规律的数组,那么可能就只有那种写法了,但是这个题目(红字)告诉我们这个数字从上到下,从左到右都是递增的,所以我们可以从这个方向入手。我们发现,我们找一个数,最快的办法无非就是利用二分进行查找,但是我们发现这个题目是一个二维单调的。所以我们要想办法找到一个方法可以在我们判断到数字小了或者大了的时候可以进行一个范围的缩小。这样的话,我们就可以定义两个指针,放到右上角或者左下角。这里我放到右上角进行讨论。
    初始的时候,我们的两个x和y的指针放到了右上角。这样的话我们可以与目标数进行比较,如果发现这个数比目标数小了,我们就把y指针往下进行移动,y指针自增。如果发现这个数比目标数大了,我们就可以往左边方向进行移动,x指针自减。

具体代码实现如下:

(右上角代码)

 bool Find(int target, vector<vector<int> > array) {int len1=array.size();  // 计算行数int len2=array[0].size();  //计算列数int posx=0,posy=len2-1; //定义两个指针,分别指向x和y的坐标while(posx<len1&&posy>=0){// 当前的数字比目标数字大,x指针左移if(array[posx][posy]>target){posy--;// 当前的数字比目标数字小,y指针往下移动}else if(array[posx][posy]<target){posx++;}else{return true;}}return false;}

(左下角代码)

bool Find(int target, vector<vector<int> > array) {int len1=array.size(); // 行数int len2=array[0].size();  //列数int posx=len1-1,posy=0; // 定义两个指针,分别表示x和y的指针while(posx>=0&&posy<len2){// 当前的数字比目标数字大,x指针往上移动if(array[posx][posy]>target){posx--;// 当前的数字比目标数字小,y指针往右方向移动}else if(array[posx][posy]<target){posy++;}else{return true;}}return false;}

复杂度分析:

由于x和y分别只能往一个方向进行移动。

  • 右上角版,最有最多只会遍历一行和一列的长度,时间复杂度为O(n+m)

  • 二维数组存储所有的元素,空间的复杂度为O(1)

我们做个总结,相对于第一题,第二题的时间复杂度从n^2到m+n得到提升,提升的原因就是很好的利用升序这个特点来优化。

二维数组中的查找(两种解法,各有千秋)相关推荐

  1. 剑指offer笔记(四)二维数组中的查找

    二维数组中的查找 题目:在一个二维数组中,每一行都按照从左到右的顺序递增,每一列都按照从上到下的顺序递增,判断一个数是否在数组中 例如: 1 2 8 9 2 4 9 12 4 7 10 13 6 8 ...

  2. 算法题001 剑指Offer 面试题三:二维数组中的查找

    剑指Offer题目1:二维数组中的查找 题目描述: http://ac.jobdu.com/problem.php?cid=1039&pid=0 在一个二维数组中,每一行都按照从左到右递增的顺 ...

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

    剑指 Offer 04. 二维数组中的查找 思路一:暴力解法,两层for循环查找 时间复杂度为:O(nm) 空间复杂度为:O(1) class Solution {public:bool findNu ...

  4. 《剑指Offer》——二维数组中的查找(JZ1)C++

    文章目录 前言 题目:JZ1 二维数组中的查找 一.暴力解法 二.优化解法 总结 前言 题目:JZ1 二维数组中的查找 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序, ...

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

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

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

    [LeetCode]剑指 Offer 04. 二维数组中的查找 文章目录 [LeetCode]剑指 Offer 04. 二维数组中的查找 package offer;public class Solu ...

  7. 剑指offer刷题 04. 二维数组中的查找

    剑指 Offer 04. 二维数组中的查找 1. 问题描述 在一个 n * m 的二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个高效的函数,输入这样的一 ...

  8. 剑指offer:面试题04. 二维数组中的查找

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

  9. php查找二维数组下标,PHP实现二维数组中的查找算法小结

    本文实例讲述了PHP实现二维数组中的查找算法.分享给大家供大家参考,具体如下: 方法1:silu从左下角最后一行的第一个元素开始,遍历.如果小于target 则遍历该行的所有元素,找到结束.如果大于继 ...

最新文章

  1. android ViewPager页面左右滑动切换
  2. 高考地理背熟这些知识可以拿80%的分数(1)
  3. httpurlconnect设置中文参数_CNC机床参数的设置及报警解除,赶紧收藏吧!
  4. 玩转lib-notify (转自PT牛的博客)
  5. ​Linux进程管理工具
  6. 2-ESP8266 SDK开发基础入门篇--点亮一个灯
  7. JAVA如何选中一行上移_js操作table中tr的顺序实现上移下移一行的效果
  8. 下周开始读《Principles of Program Analysis》
  9. python同时输出名字和时间_Python练习小工具——根据Exif的拍摄时间和设备名批量重命名照片...
  10. 整理了一些大数据资料与电子书,免费分享给大家!!
  11. 游程编码解密(C语言详解)
  12. 英语单词记忆 词源法-思维导图 序
  13. 线上插画培训班有用吗,教你选靠谱的插画课程
  14. 测试REST Web服务
  15. 计算机视觉 (CV) 相关的会议或期刊投稿时间
  16. excel自学第1天_excel制作项目时间进度表_excel表头
  17. 并发与多线程相关知识点梳理
  18. 微信开放平台 错误码61007: api is unauthorized to component
  19. JavaScript实现弹出“确定/取消”对话框的方法
  20. 田志刚:卓越绩效评价中的知识管理模块分析

热门文章

  1. 无需PS,这些工具也能轻松更换寸照底色
  2. 官网下载git缓慢问题
  3. Python Fitter 判断数据样本的分布函数拟合
  4. 怎么提升效率?衡量指标又是是什么?
  5. Windows搭建青龙面板教程
  6. Java读取hdfs文件权限问题
  7. linux打印文件名称唯美,程序员的情人节应该这么优雅度过(附源码)
  8. 计算机程序占用端口,程序启动发现端口被占?3步查出它是谁!
  9. 百度地图 开启 绘制 功能(画圆)
  10. 追寻红色记忆、晋国文明之思考毕业季研学营