Javascript算法——二分查找

1.数组

1.1二分查找

1.搜索索引

开闭matters!!![left,right]与[left,right)

/**
 * @param {number[]} nums
 * @param {number} target
 * @return {number}
 */
var search = function(nums, target) {
    let left=0;
    let right=nums.length-1;
    //[left,right],相等时能取到,有意义
    while(left<=right){
        let mid =Math.floor((left+right)/2);
        if(target===nums[mid]){
            return mid;
        }else if (target>nums[mid]) {
            left=mid+1;
        }else{
            right=mid-1;
        }
    }
    return -1;

};
console.log(search([-1,0,3,5,9,12],2))//-1
console.log(search([-1,0,3,5,9,12],2))//4

                                                                       VS 

/**
 * @param {number[]} nums
 * @param {number} target
 * @return {number}
 */
var search = function(nums, target) {
    // right是数组最后一个数的下标+1,nums[right]不在查找范围内,是左闭右开区间
    let mid, left = 0, right = nums.length;    
    // 当left=right时,由于nums[right]不在查找范围,所以不必包括此情况
    while (left < right) {
        // 位运算 + 防止大数溢出
        mid = left + ((right - left) >> 1);
        // 如果中间值大于目标值,中间值不应在下次查找的范围内,但中间值的前一个值应在;
        // 由于right本来就不在查找范围内,所以将右边界更新为中间值,如果更新右边界为mid-1则将中间值的前一个值也踢出了下次寻找范围
        if (nums[mid] > target) {
            right = mid;  // 去左区间寻找
        } else if (nums[mid] < target) {
            left = mid + 1;   // 去右区间寻找
        } else {
            return mid;
        }
    }
    return -1;
};
 2.搜索插入位置

/**
 * @param {number[]} nums
 * @param {number} target
 * @return {number}
 */
var searchInsert = function(nums, target) {
    let left=0;
    let right=nums.length-1;
    //[left,right],相等时能取到,有意义
    while(left<=right){
        let mid =Math.floor((left+right)/2);
        if(target===nums[mid]){
            return mid;
        }else if (target>nums[mid]) {
            left=mid+1;
        }else{
            right=mid-1;
        }
    }
      // 分别处理如下四种情况
      // 目标值在数组所有元素之前  [0, -1]
      // 目标值等于数组中某一个元素  return middle;
      // 目标值插入数组中的位置 [left, right],return  right + 1
      // 目标值在数组所有元素之后的情况 [left, right],这是右闭区间,所以  return right + 1

    return right+1;

};
console.log(search([1,3,5,6],0))//0
console.log(search([1,3,5,6],3))//1
console.log(search([1,3,5,6],4))//2
console.log(search([1,3,5,6],7))//4

其余三种都可以归纳为right+1 

3.在排序数组中查找元素的第一个和最后一个位置

  • 找左边界时,需将right赋给左边界,所以在target<=num[mid]时更新right并更新左边界
  • 找右边界时,需将left赋给右边界,所以在target>=num[mid]时更新left并更新右边界
  • 情况二,通过rightBorder-leftBorder>1条件判断
var searchRange = function(nums, target) {
    const getLeftBorder = (nums, target) => {
        let left = 0, right = nums.length - 1;
        let leftBorder = -2;// 记录一下leftBorder没有被赋值的情况
        while(left <= right){
            let middle = left + ((right - left) >> 1);
            if(nums[middle] >= target){ // 寻找左边界,nums[middle] == target的时候更新right
                right = middle - 1;
                leftBorder = right;
            } else {
                left = middle + 1;
            }
        }
        return leftBorder;
    }

    const getRightBorder = (nums, target) => {
        let left = 0, right = nums.length - 1;
        let rightBorder = -2; // 记录一下rightBorder没有被赋值的情况
        while (left <= right) {
            let middle = left + ((right - left) >> 1);
            if (nums[middle] > target) {
                right = middle - 1;
            } else { // 寻找右边界,nums[middle] == target的时候更新left
                left = middle + 1;
                rightBorder = left;
            }
        }
        return rightBorder;
    }

    let leftBorder = getLeftBorder(nums, target);
    let rightBorder = getRightBorder(nums, target);
    // 情况一
    if(leftBorder === -2 || rightBorder === -2) return [-1,-1];
    // 情况三
    if (rightBorder - leftBorder > 1) return [leftBorder + 1, rightBorder - 1];
    // 情况二
    return [-1, -1];
};
4.X的平方根
function mySqrt(x) {  
    if (x === 0) return 0; // 特殊情况处理:0的平方根是0  
  
    let left = 1; // 搜索范围的左边界  
    let right = Math.floor(x / 2) + 1; // 搜索范围的右边界,x/2是一个合理的上限,因为平方根不会超过x/2(对于非负整数x)  
  
    while (left <= right) {  
        let mid = Math.floor((left + right) / 2); // 计算中间值  
        let square = mid * mid; // 计算中间值的平方  
  
        if (square === x) {  
            return mid; // 如果平方正好等于x,直接返回  
        } else if (square < x) {  
            left = mid + 1; // 如果平方小于x,说明平方根在mid的右侧,移动左边界  
        } else {  
            right = mid - 1; // 如果平方大于x,说明平方根在mid的左侧或正好是mid(但我们需要整数部分,所以向左移动)  
        }  
    }  
  
    // 循环结束时,left会指向比实际平方根大的最小整数,而right会指向比实际平方根小的最大整数  
    // 因为我们需要整数部分,所以返回right(它是最后一个使得mid*mid <= x的mid值)  
    return right;  
}  
  
// 测试  
console.log(mySqrt(4));  // 输出: 2  
console.log(mySqrt(8));  // 输出: 2 (8的平方根约为2.8284,取整数部分2)  
console.log(mySqrt(15)); // 输出: 3 (15的平方根约为3.8729,取整数部分3)

解释

  1. 边界条件
    • 如果 x 为0,则直接返回0。
  2. 搜索范围
    • 左边界 left 初始化为1,因为0的平方根是0(已经特殊处理),而任何正数的平方根至少为1。
    • 右边界 right 初始化为 Math.floor(x/2)+1,因为平方根不会超过 x/2(对于非负整数 x)。加1是为了确保在 x 为完全平方数时能够包含这个平方根。
  3. 二分查找
    • 在每次迭代中,计算中间值 mid 及其平方 square
    • 根据 square 与 x 的比较结果,移动左边界或右边界。
  4. 返回结果
    • 循环结束时,返回 right,它是最后一个使得 mid * mid <= x 的 mid 值,也就是我们要找的平方根的整数部分。

这种方法的时间复杂度是 O(logn),其中 n 是 x 的值,因为每次迭代都会将搜索范围减半。

更精确 (待进一步补充)

function mySqrt(x) {  
    if (x === 0) return 0; // 特殊情况处理:0的平方根是0  
  
    let guess = x; // 初始猜测值设为x本身(对于非负整数,平方根不会超过x本身)  
    let epsilon = 1; // 精度控制,用于判断迭代是否结束  
  
    while (Math.abs(guess * guess - x) >= epsilon) {  
        // 牛顿迭代公式:guess = (guess + x / guess) / 2  
        guess = Math.floor((guess + Math.floor(x / guess)) / 2);  
        // 为了确保精度,逐步减小epsilon  
        epsilon /= 10;  
    }  
  
    return guess;  
}  
  
// 测试  
console.log(mySqrt(4));  // 输出: 2  
console.log(mySqrt(8));  // 输出: 2 (8的平方根约为2.8284,取整数部分2)  
console.log(mySqrt(15)); // 输出: 3 (15的平方根约为3.8729,取整数部分3)
  1. 初始猜测值
    • 对于非负整数 x,初始猜测值设为 x 本身,因为平方根不会超过 x 本身。
  2. 牛顿迭代公式
    • 牛顿迭代法的公式为:new_guess=(old_guess+x/old_guess)/2​​
    • 这个公式通过不断迭代来逼近平方根的值。
  3. 精度控制
    • 使用 epsilon 来控制精度,初始设为 1。
    • 每次迭代后,将 epsilon 除以 10,逐步减小精度要求,确保最终结果的准确性。
  4. 取整
    • 使用 Math.floor 函数来确保结果只保留整数部分。
 5.有效的完全平方数(与上类似)
/**
 * @param {number} num
 * @return {boolean}
 */
var isPerfectSquare = function(num) {
    if(num===1)return true
    let left=1;
    let right=Math.floor(num/2)+1;
    //天天天,你条件写错了!!!!
    while(left<=right){
        let mid = Math.floor((left+right)/2);
        let square=mid*mid;
        if(square===num){
            return true;
        }else if(square>num){
            right=mid-1;
        }else{
            left=mid+1;
        }
    }
    return false;
};
  1. 题目:给定一个n个元素有序的(升序)整型数组nums和一个目标值target,写一个函数搜索nums中的target,如果目标值存在返回下标,否则返回-1。
    • 解析:这是二分查找算法的最基本应用。通过设定左右指针,不断缩小搜索范围,直到找到目标值或确定目标值不存在。
  2. 题目:给定一个按照非递减顺序排列的整数数组nums,和一个目标值target。请你找出给定目标值在数组中的开始位置和结束位置。如果数组中不存在目标值target,返回[-1, -1]。
    • 解析:这个问题可以先用二分查找找到目标值的一个位置,然后通过双指针从中间向两边扩散,找到目标值的开始位置和结束位置。这种方法的时间复杂度为O(log n + k),其中n是数组的长度,k是目标值在数组中出现的次数。
  3. 题目:在旋转排序数组中查找目标值(假设数组中不存在重复元素)。
    • 解析:旋转排序数组是指一个递增排序数组经过一次旋转得到的数组。这个问题可以通过修改二分查找算法来解决。首先,找到数组中的“旋转点”(即数组从递增变为递减的点),然后根据目标值与旋转点的大小关系,在数组的左侧或右侧进行二分查找。
  4. 题目:在有序数组中查找第一个大于给定值的元素。
    • 解析:这个问题可以通过二分查找算法来解决。在每次迭代中,根据中间元素与目标值的大小关系,更新搜索范围,直到找到第一个大于目标值的元素或确定不存在这样的元素。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/898901.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

大话网络协议:从OSI七层模型说开去

时至今日,互联网已经是大家日常生活中不可或缺的一部分,购物、点餐、刷剧、网课,已经融入了我们生活的方方面面。但网络具体是怎么工作的呢? 特别是我们具体从事软件研发、ICT行业的同学,理解和掌握这个我们产品运行的基础设施尤为必要。 本文,我们会力争用最简单易懂的…

秋季猫咪疯狂掉毛,宠物空气净化器有用吗?性价比高的该怎么选?

我家猫真的是换季就变掉毛怪&#xff0c;整只猫“虚胖”了一大圈不止&#xff0c;在阳光下可以看见非常多飘在空气中的浮毛。浮毛到处乱飞&#xff0c;沉积在黑色的衣服上&#xff0c;就形成白色的薄膜。自从养猫后&#xff0c;我再也没穿过深色的衣服。 现在每天都给它梳毛&am…

Linux文件的查找和打包以及压缩

文件的查找 文件查找的用处&#xff0c;在我们需要文件但却又不知道文件在哪里的时候 文件查找存在着三种类型的查找 1、which或whereis&#xff1a;查找命令的程序文件位置 2、locate&#xff1a;也是一种文件查找&#xff0c;但是基于数据库的查找 3、find&#xff1a;针…

Vue.js 学习总结(9)—— Vue 3 组件封装技巧

1、需求说明 需求背景&#xff1a;日常开发中&#xff0c;我们经常会使用一些UI组件库诸如and design vue、element plus等辅助开发&#xff0c;提升效率。有时我们需要进行个性化封装&#xff0c;以满足在项目中大量使用的需求。错误示范&#xff1a;基于a-modal封装一个自定…

【AIGC半月报】AIGC大模型启元:2024.10(下)

【AIGC半月报】AIGC大模型启元&#xff1a;2024.10&#xff08;下&#xff09; (1) Janus&#xff08;两面神&#xff09;&#xff08;DeepSeek 1.3B多模态大模型&#xff09;(2) Stable Diffusion 3.5&#xff08;StabilityAI文生图大模型&#xff09;(3) Mochi 1&#xff08;…

Python文件操作(读取、写入、修改和删除)

目录 一、文件的读取 二、文件的写入 三、文件的修改 四、文件的删除 Python是一种功能强大的编程语言&#xff0c;文件操作是编程中常见的需求。本文将详细介绍Python中的文件操作&#xff0c;包括文件的读取、写入、修改和删除&#xff0c;帮助读者掌握Python文件操作的基…

分布式系统之异步与消息队列(MQ)(原理+代码实战一文讲清!)

异步 什么是异步 异步编程是一种编程范式&#xff0c;它允许程序在等待操作完成&#xff08;如等待网络响应、文件读写等&#xff09;时继续执行其他任务。这种编程方式对于提高程序的性能和响应性至关重要&#xff0c;尤其是在处理耗时操作或在资源受限的环境中。下面我将更…

山东以“八策并举”确保人民满意学前教育“普惠落地”

10月19日-22日&#xff0c;2024年中国学前教育研究会学术年会在山东国际会展中心召开。年会围绕“优质普惠可持续——加强学前教育高质量发展的法治保障”主题&#xff0c;通过5场主旨报告、28个园所观摩、10个分论坛交流研讨&#xff0c;为2200余名嘉宾提供智慧盛宴。成为近年…

URP学习四

一.Bilt To RTHandle feature代码&#xff1a; 二.DistortTunnel 只有个飞机却有很多太空场景。因为设置了其他pass来渲染背景 队列添加3个Pass&#xff1a; 第一个Pass把颜色图进行输出 第二个Pass&#xff1a;创建了个纹理 加了个扰动&#xff0c;把纹理进行输出 第三个pas…

Postman使用-基础篇

前言 本教程将结合业界广为推崇和使用的RestAPI设计典范Github API&#xff0c;详细介绍Postman接口测试工具的使用方法和实战技巧。 在开始这个教程之前&#xff0c;先聊一下为什么接口测试在现软件行业如此重要&#xff1f; 为什么我们要学习Postman&#xff1f; 现代软件…

电子木鱼小游戏小程序源码系统 带完整的安装代码包以及搭建部署教程

系统概述 在快节奏的生活中&#xff0c;人们越来越注重内心的平静与放松。电子木鱼小游戏小程序正是基于这一需求而诞生的&#xff0c;它将传统的木鱼文化与现代科技相结合&#xff0c;为用户提供了一个简单、方便、有趣的冥想与放松工具。通过敲击屏幕上的虚拟木鱼&#xff0…

Windows 下 golang 多版本管理

三年前的旧文&#xff0c;最新要切版本&#xff0c;翻了出来&#xff0c;现在依然有用&#xff0c;分享出来~ 当前 golang 的各个版本还有些不兼容的问题&#xff0c;最近遇到 go-micro 框架只能运行在 go1.13~1.14 的版本情况&#xff0c;而我本地 windows 环境安装的 Golang …

C++ [项目] 愤怒的小鸟

现在才发现C游戏的支持率这么高&#xff0c;那就发几篇吧 零、前情提要 此篇为 制作,由于他没有CSDN,于是由我代发 一、基本介绍 支持Dev-C5.11版本(务必调为英文输入法),基本操作看游戏里的介绍,怎么做的……懒得说,能看懂就看注释,没有的自己猜,如果你很固执……私我吧 …

蘑菇书(EasyRL)学习笔记(1)

1、强化学习概述 强化学习&#xff08;reinforcement learning&#xff0c;RL&#xff09;讨论的问题是智能体&#xff08;agent&#xff09;怎么在复杂、不确定的环 境&#xff08;environment&#xff09;里面去最大化它能获得的奖励。如下图所示&#xff0c;强化学习…

huggingface的数据集下载(linux下clone)

1. 安装lfs sudo apt-get install git-lfs 或者 apt-get install git-lfs 2. git lfs install git lfs install 3. git clone dataset包 第2&#xff0c;3步骤的截图如下&#xff1a;

Kubernetes学习笔记

Kubernetes学习笔记 API格式前缀API组API版本 Pod概念优势局限性创建Pod ReplicationController概念配置Pod模板 Kubernetes架构概述节点定义管理节点名称唯一性节点自注册手动节点管理节点状态节点心跳节点控制器逐出速率限制资源容量跟踪 API Kubernetes把其管理的资源均视为…

现代数字信号处理I-P4 CRLB+LMMSE 学习笔记

目录 学习资料视频链接&#xff1a; 1. 估计参数的CRLB回顾 2. 参数变换下的CRLB拓展 3. 矢量参数下的CRLB扩展 3.1 矢量参数下的CRLB公式 3.2 两个矩阵不等式关系的意义说明 3.3 矢量参数下CRLB公式的证明过程 4. 线性估计 重点注意事项&#xff1a;此处的线性估计&am…

零磁通电流探头的原理

在电力电子和自动化控制领域&#xff0c;电流测量的准确性至关重要。传统的开环式电流探头&#xff0c;尽管在交流电流测量中表现出色&#xff0c;但在直流或大电流测量时&#xff0c;常面临磁芯饱和、剩磁及温度变化带来的测量误差问题。为此&#xff0c;零磁通电流探头&#…

​​Spring6梳理17——基于XML的自动装配

以上笔记来源&#xff1a; 尚硅谷Spring零基础入门到进阶&#xff0c;一套搞定spring6全套视频教程&#xff08;源码级讲解&#xff09;https://www.bilibili.com/video/BV1kR4y1b7Qc 目录 ①引入 ②场景模拟 2.1 创建UserController类文件 2.2 创建UserService接口文件 2…

同济大学计算机考研

文章目录 一、初试1.院校情况1.复试名单2.报录比3.学硕人数 二、复试(一) 数据库2016复试题一、选择题 &#xff08;Multiple Choices&#xff09;二、简答题 2018复试题一、选择题&#xff08;一&#xff09;数据库&#xff1a;1-10&#xff08;二&#xff09;C语言&#xff1…