1389 蓝桥杯 二分查找数组元素 简单

1389 蓝桥杯 二分查找数组元素 简单

cpp 复制代码
//C++风格解法1,lower_bound(),通过率100%
//利用二分查找的方法在有序的数组中查找,左闭右开
#include <bits/stdc++.h>
using namespace std;

int main(){
    
    int data[200];
    for(int i = 0 ; i < 200 ; ++i) data[i] = 4 * i + 6;
    int tager; cin >> tager;    //输入目标值
    cout << lower_bound(data,data + 200,tager) - data;

    return 0;
}

在有序数组中进行二分查找,升序,查找第一个 >= target的元素,时间复杂度O(logn)

lower_bound(data,data + 200,tager)返回物理地址,减去首地址得到下标

cpp 复制代码
// 升序数组中:
upper_bound(a.begin(), a.end(), x); // 查找第一个 > x的元素
lower_bound(a.begin(), a.end(), x); // 查找第一个 >= x的元素
// 降序数组中:
upper_bound(a.begin(), a.end(), x, greater<type>()); // 查找第一个 < x的元素
lower_bound(a.begin(), a.end(), x, greater<type>()); // 查找第一个 <= x的元素

排序的时候,默认是从小到大,但是第三个参数用greater会变成从大到小,而不需要cmp

upper_bound默认是找大于,但是第三个参数用greater就是找小于,lower_bound同理可得

cpp 复制代码
//C风格解法2,逆推,虽然用了cin、cout,通过率100%
#include <iostream>
using namespace std;
int main(){
  ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
  int x;
  cin >> x;
  cout << (x - 6) / 4 << endl;
  return 0;
}
cpp 复制代码
//C风格解法3,二分法左闭右开
#include <iostream>
using namespace std;

int main(){
  ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
  int x;cin >> x;
  int data[200];
  for(int i = 0 ; i < 200 ; i ++)data[i] = 4 * i + 6;
  
  int left = 0,right = 200;   //定义target在左闭右开的区间里,即[left, right)
  //right指向最后一个元素的后一个元素
  while(left < right){    //如果一开始left = right,target在[left, right)区间上无意义
    int mid = left + ((right - left) >> 1);   //等同于 (left + right) / 2,防止溢出
    if(data[mid] >= x)right = mid;    //target在左区间[left, mid] 
    else left = mid + 1;    //target在右区间[left, right) 
  }    

  cout << left << endl;   //left = right
  return 0;
}

基于此方法改动可能会超时,

如传统三段式if、else if、else,二分法左闭右闭,while(left <= right)等

right = left时,循环条件被修改成 while (left <= right) 会接着做循环

出错,如data[mid] > x,注意返回的不是mid

出错,声明mid,输出mid

如果是左闭右闭,left = 0,right = 199, while (left <= right)

定义了target在左闭右闭的区间内,[left, right],right指向最后一个元素

left = right时,target在区间[left, right]仍然有效

若循环条件修改为while (left < right),nums[middle] = A 时< target = B,

此时 left = mid + 1,left = right,而循环条件为while (left < right),

还未找到A 的情况下算法就跳出了循环

reference:

彻底记住 lower_bound 和 upper_bound 功能和用法_lower_bound和upper_bound的用法-CSDN博客

关于lower_bound( )和upper_bound( )的常见用法_lowerbound和upperbound-CSDN博客

lower_bound, upper_bound, greater, less 用法_lower_bound greater-CSDN博客

【二分查找】详细图解_二分查找法流程图-CSDN博客

图文并茂带你入门二分查找算法 - 知乎 (zhihu.com)

相关推荐
@小码农4 分钟前
6547网:202512 GESP认证 C++编程 一级真题题库(附答案)
java·c++·算法
TDengine (老段)10 分钟前
TDengine C/C++ 连接器入门指南
大数据·c语言·数据库·c++·物联网·时序数据库·tdengine
自然语16 分钟前
人工智能之数字生命-特征类升级20260106
人工智能·算法
菜鸟233号17 分钟前
力扣343 整数拆分 java实现
java·数据结构·算法·leetcode
赫凯21 分钟前
【强化学习】第五章 时序差分算法
算法
vyuvyucd22 分钟前
C++ vector容器完全指南
c++
liulilittle22 分钟前
XDP VNP虚拟以太网关(章节:三)
网络·c++·网络协议·信息与通信·通信·xdp
leiming625 分钟前
c++ find_if 算法
开发语言·c++·算法
yuanmenghao26 分钟前
自动驾驶中间件iceoryx - 内存与 Chunk 管理(三)
数据结构·c++·算法·链表·中间件·自动驾驶
_OP_CHEN32 分钟前
【算法基础篇】(四十三)数论之费马小定理深度解析:从同余性质到乘法逆元
c++·算法·蓝桥杯·数论·acm/icpc