二分查找(一)

二分查找看似简单,但是有很多的细节要注意。

题目是牛客NC105,找到有序数组中第一个大于或者等于所查找的数字。

初步写了如下的代码:

class Solution {
public:
/**
* 二分查找
* @param n int整型 数组长度
* @param v int整型 查找值
* @param a int整型vector 有序数组
* @return int整型
*/
int upper_bound_(int n, int v, vector<int>& a) {
// write code here
int mid=0;
int left=0;
int right=n-1;
while(left < right)
{ //mid = left+(right-left)/2;
mid = (left+right)/2;
if(a[mid] > v){
if(a[mid-1]<v) return mid;
else{
right = mid-1;
} }else if(a[mid] < v){
left = mid+1;
}else if(a[mid] == v)
{
while(a[mid] == v){
mid--;
}
return mid+1;//这里必须是mid+1?
}
}
return n+1;
} };

运行之后发现测试用例不能完全通过,这段代码有什么问题?

首先是mid = left + (right-left)/2, 这和mid = (left+right)/2相比,有什么区别?

实际测试的时候,没有区别,但是一旦left和right都很大的时候,则有可能会造成数组越界。

因此,使用(right-left)/2可以很大程度避免两个很大的数相加导致数组越界的情况发生。

其次是把right=mid写成了right=mid-1。为什么?

因为a[mid]>v的情况下,a[mid]是符合题目要求的大于所查找数的,所以,right=mid,而不是right=mid-1。也就是说,没有把它排除。

代码还有bug,

while(a[mid] == v)
{
mid--;
}
return mid+1;

把return mid+1改为了return mid+2,因为输出是按1而不是按0开始的。

我的解题思路是,如果是大于v,则判断其左侧的位置是否小于v,如果小于则是第一个,如果等于v,则不断向左移动,直到找到小于v的位置。

标准题解:

标准题解把a[mid]>=v放到一起作为判断条件。这是因为既然是大于等于,则a[mid]>=v都是符合要求的,只要判断是不是第一次即可。

什么是第一次呢? 那就是mid左边的那个数如果是小于v的,则mid就是符合要求的。二分查找的本质,是left在等于right之前,不断查找符合条件的位置。

mid发现的位置是中间位置而不是'第一次'出现的位置。因为大于等于所找的值,这本身就是符合要求的条件,二分查找找的就是符合某条件的位置,不断地通过二分查找直到找到。

因此 a[mid]>=v可以写在一起。

最后经过优化的代码:

int upper_bound_(int n, int v, vector<int>& a) {
// write code here
int mid=0;
int left=0;
int right=n-1;
while(left < right)
{ mid = left+(right-left)/2;//为什么要这样写Mid?
if(a[mid] > v){
if( mid == 0|| a[mid-1]<v) return mid+1;
else{
right = mid;
}
}else if(a[mid] < v){
left = mid+1;
}else if(a[mid] == v)
{
if(a[mid-1]<v)
return mid+1;
else right = mid;
}
}
return n+1;
}

如果是left <=right呢?修改之后仍不会影响结果。

做完该题后,再看leetcode 704

class Solution {
public:
int search(vector<int>& nums, int target) {
int len = nums.size();
int left = 0;
int right = len-1; int mid = 0; while(left < right)//如果只有一个元素会如何呢?
{
mid = left + (right-left)/2; if(nums[mid] == target){
return mid;
}else if(nums[mid] > target)
{
right = mid - 1;
}else if(nums[mid] < target)
{
left = mid + 1;
}
}
return -1;
}
};

这段代码无法通过全部的测试用例,比如在测试元素只有一个的数组的时候报错了,因此要怎么办?

最好是用while(left<=right),这样就避免了一个元素时不进入循环的情况了。

继续趁热打铁,看leetcode 74题,搜索二维矩阵。

获取二维数组的行和列

行数 = sizeof(array)/sizeof(array[0])
列数 = sizeof(array[0])/sizeof(array[0][0])

行数计算方法就是用二维数组的整个size除以第一行的size

列数计算方法就是第一行除以每个元素的size.

求vector二维数组的行数,直接就是size()方法。

class Solution {
public:
bool searchMatrix(vector<vector<int>>& matrix, int target) {
//只要先按照第一列进行二分查找,找到第一个大于等于的位置,然后该位置的
//上一个数组就可能存在一个目标值,在在数组中继续二分查找。 //行数
//int m = sizeof(matrix)/sizeof(matrix[0]);
int m = matrix.size();//直接就是行数
int up = 0;
int down = m-1;
int mid1=0; while(up <= down)
{
mid1 = up+(down-up)/2; if(matrix[mid1][0] == target){
return true;
}else if(matrix[mid1][0] > target){
if(matrix[mid1-1][0] < target){//上一行中必然存在,mid1-1行可能存在
break;
}else{
down = mid1 -1 ;
} }else if(matrix[mid1-1][0] < target){
up = mid1 + 1;
}
} int n = matrix[0].size();
int left = 0;
int right = n-1;
int mid2 = 0;
while(left <= right)
{
mid2 = left+(right-left)/2; if(matrix[mid1-1][mid2] == target){
return true;
}else if(matrix[mid1-1][mid2] > target)
{
right = mid2-1;
}else if(matrix[mid1-1][mid2] < target)
{
left = mid2+1;
}
} return false; }
};

以上代码有什么错误?

没有考虑到空参数的问题,会出现'runtime error: reference binding to null pointer of type'提示。

如果是[[]] , 二维数组的特殊情形, 完全空,只有一行一列,多行一列,一行多列。这四种情况,都要在运行的时候考虑到。

因此,最稳妥的方式就是把二维数组转化为一维数组。但是一维数组有没有最大长度? 有,vector是有最大长度的。https://blog.csdn.net/iceboy314159/article/details/80329979

但是在本题中足够。

标准题解:

也不需要先将二维放入一维徒增空间复杂度,而是直接把二维数组当成一维数组。

a[mid]转换就是b[mid/n][mid % n],为什么? 第k个数是在二维数组种的位置便是[k / n][k % n]

这里的k包括了从0位置开始的记录。

class Solution {
public:
bool searchMatrix(vector<vector<int>>& matrix, int target) {
//转化为一维数组,但是要考虑到长度问题。 int m = matrix.size();
int n = matrix[0].size(); if( m == 0 || n == 0) return false; int left = 0;
int right = m*n-1; int mid = 0; while(left <= right)
{
mid = left+(right-left)/2;
int tmp = matrix[mid/n][mid%n];
if(tmp > target){
right = mid - 1;
}else if(tmp < target)
{
left = mid + 1;
}else
return true;
} return false;
}
};

这一块代码为什么还有问题? 输入了[]之后,还是会有错误?

因为输入了[]之后,其中这不是二维数组,所以行数为0,而行数为0的情况下判断martix[0].size()就是不可能的。

所以应该先判断行数是否存在,存在后再计算列数。

改代码为:

class Solution {
public:
bool searchMatrix(vector<vector<int>>& matrix, int target) {
//转化为一维数组,但是要考虑到长度问题。 int m = matrix.size();
if(m == 0) return false;
int n = matrix[0].size(); if( n == 0) return false; int left = 0;
int right = m*n-1; int mid = 0; while(left <= right)
{
mid = left+(right-left)/2;
int tmp = matrix[mid/n][mid%n];
if(tmp > target){
right = mid - 1;
}else if(tmp < target)
{
left = mid + 1;
}else
return true;
} return false;
}
};

当然,这样做的时间复杂度是MN,而空间复杂度为1.有没有更好的方法呢? 那就是前面提到的先判断在哪一行,再判断是该行的哪一列。

加上了对二维数组的判空后,第一种做法仍然会报错,这是由于还有1行1列,1行多列的情况存在,这种情况下,mid-1就会出现负值,导致出错。

这样的话,判断条件非常多,所以不选择在第一列找,而是在最后一列找,在最后一列找到第一次大于target的,则找到的mid就是可能存在该数的行。

二分查找中遇到Mid==0该怎么办

mid为0的情况,比如前一个mid为1,但是还是不满足条件,然后再次除以2,从而mid==0,此时left和right也相等,则不需要再查找,而是直接跳出。

还有一种情况,就是只有一个元素的数组。总之,mid为0表示之后不需要再次计算。

NC105 二分查找法的更多相关文章

  1. jvascript 顺序查找和二分查找法

    第一种:顺序查找法 中心思想:和数组中的值逐个比对! /* * 参数说明: * array:传入数组 * findVal:传入需要查找的数 */ function Orderseach(array,f ...

  2. 用c语言编写二分查找法

    二分法的适用范围为有序数列,这方面很有局限性. #include<stdio.h> //二分查找法 void binary_search(int a[],int start,int mid ...

  3. java for循环和数组--冒泡排序、二分查找法

    //100以内与7相关的数   for(int a=1;a<=100;a++){    if(a%7==0||a%10==7||a/10==7){     System.out.print(a+ ...

  4. 二分查找法 java

    前几天去面试,让我写二分查找法,真是哔了狗! 提了离职申请,没事写写吧! 首先二分查找是在一堆有序的序列中找到指定的结果. public class Erfen { public static int ...

  5. 学习练习 java 二分查找法

    package com.hanqi; import java.util.*; public class Test5 { public static void main(String[] args) { ...

  6. Java-数据结构与算法-二分查找法

    1.二分查找法思路:不断缩小范围,直到low <= high 2.代码: package Test; import java.util.Arrays; public class BinarySe ...

  7. 选择、冒泡排序,二分查找法以及一些for循环的灵活运用

    import java.util.Arrays;//冒泡排序 public class Test { public static void main(String[] args) { int[] ar ...

  8. R语言实现二分查找法

    二分查找时间复杂度O(h)=O(log2n),具备非常高的效率,用R处理数据时有时候需要用到二分查找法以便快速定位 Rbisect <- function(lst, value){ low=1 ...

  9. java学习之—递归实现二分查找法

    /** * 递归实现二分查找法 * Create by Administrator * 2018/6/21 0021 * 上午 11:25 **/ class OrdArray{ private lo ...

随机推荐

  1. Linux centos7 nginx 平滑升级

    2021-08-19为了方便读者的阅读,该文通篇使用绝对路径,各位朋友们在实际上操作中可以根据实际情况编写路径(#^.^#)1. 当前环境 # system cat /etc/redhat-relea ...

  2. JavaScript高级程序设计(读书笔记)之函数表达式

    定义函数的方式有两种:一种是函数声明,另一种就是函数表达式. 函数声明的一个重要特征就是函数声明提升(function declaration hoisting),意思是在执行代码前会先读取函数声明. ...

  3. noip模拟17

    \(\color{white}{\mathbb{霞光划破暗淡天际,月影彷徨,鸡鸣仿佛,冀之以继往开来,名之以:黎明}}\) 今天似乎取得了有史以来最好的成绩~ 前两名都 A 掉了 \(t3\),然鹅 ...

  4. Kubernetes环境Traefik部署与应用

    本作品由Galen Suen采用知识共享署名-非商业性使用-禁止演绎 4.0 国际许可协议进行许可.由原作者转载自个人站点. 概述 本文用于整理基于Kubernetes环境的Traefik部署与应用, ...

  5. Java Web下MySQL数据库的增删改查(二)

    前文:https://www.cnblogs.com/Arisf/p/14095002.html 在之前图书管理系统上做了改进优化 图书管理系统v2 首先是项目结构: 1.数据库的连接: 1 pack ...

  6. Linux系列(11) - PATH环境变量

    前言 在Liunx当中要想执行"执行脚本"."执行文件"或者"执行命令",需要用绝对路径:因此环境变量就出来了,将路径放到环境变量中,环境变 ...

  7. 我在学习Blazor当中踩的巨坑!Blazor WebAssembly调试

    最近嘛,看看Blazor已经蛮成熟的.顺便想在自家的框架里使用这个东西,毕竟我还是很念旧的,而且Blazor的技术栈也不麻烦.然后呢,在调试这一关我可是踩了大坑. 我的VS是2019,很早以前装的.然 ...

  8. Kafka分区策略

    Kafka分区策略 所谓分区策略是决定生产者将消息发送到哪个分区的算法.Kafka 为我们提供了默认的分区策略,同时它也支持你自定义分区策略. 常见的分区策略包含以下几种:轮询策略.随机策略 .按消息 ...

  9. 浅谈语音质量保障:如何测试 RTC 中的音频质量?

    日常音视频开会中我们或多或少会遭遇这些场景:"喂喂喂,可以听到我说话吗?我听你的声音断断续续的","咦,我怎么可以听到回声?","太吵啦,我听不清楚你 ...

  10. NOI.AC#2266-Bacteria【根号分治,倍增】

    正题 题目链接:http://noi.ac/problem/2266 题目大意 给出\(n\)个点的一棵树,有一些边上有中转站(边长度为\(2\),中间有一个中转站),否则就是边长为\(1\). \( ...