我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java和C++如何在排序数组中查找数字出现的次数

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java和C++如何在排序数组中查找数字出现的次数

这篇文章主要介绍“Java和C++如何在排序数组中查找数字出现的次数”,在日常操作中,相信很多人在Java和C++如何在排序数组中查找数字出现的次数问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java和C++如何在排序数组中查找数字出现的次数”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

1、题目

统计一个数字在排序数组中出现的次数。

示例 1:

输入: nums = [5,7,7,8,8,10], target = 8

输出: 2

示例 2:

输入: nums = [5,7,7,8,8,10], target = 6

输出: 0

提示:

  • 0 <= nums.length <= 10^5

  • -10^9 <= nums[i] <= 10^9

  • nums 是一个非递减数组

  • -10^9 <= target <= 10^9

2、思路

Java和C++如何在排序数组中查找数字出现的次数

统计一个数字在排序数组中出现的次数。

样例:

Java和C++如何在排序数组中查找数字出现的次数

如样例所示,nums = [5,7,7,8,8,10],target = 8,8在数组中出现的次数为2,于是最后返回2。

数组有序,因此可以使用二分来做。两次二分,第一次二分查找第一个>= target的位置begin;第二次二分查找最后一个<= target的位置end,查找成功则返回end - begin + 1,即为数字在排序数组中出现的次数,否则返回0,表示该数没有在数组中出现。

二分模板:

模板1

当我们将区间[l, r]划分成[l, mid]和[mid + 1, r]时,其更新操作是r = mid或者l = mid + 1,计算mid时不需要加1,即mid = (l + r)/2。

C++/java代码模板:

int bsearch_1(int l, int r){    while (l < r)    {        int mid = (l + r)/2;        if (check(mid)) r = mid;        else l = mid + 1;    }    return l;}

模板2

当我们将区间[l, r]划分成[l, mid - 1]和[mid, r]时,其更新操作是r = mid - 1或者l = mid,此时为了防止死循环,计算mid时需要加1,即mid = ( l + r + 1 ) /2。

C++/java 代码模板:

int bsearch_2(int l, int r){    while (l < r)    {        int mid = ( l + r + 1 ) /2;        if (check(mid)) l = mid;        else r = mid - 1;    }    return l;}

为什么两个二分模板的mid取值不同?

对于第二个模板,当我们更新区间时,如果左边界l更新为l = mid,此时mid的取值就应为mid = (l + r + 1)/ 2。因为当右边界r = l + 1时,此时mid = (l + l + 1)/2,相当于下取整,mid为l,左边界再次更新为l = mid = l,相当于没有变化。while循环就会陷入死循环。因此,我们总结出来一个小技巧,当左边界要更新为l = mid时,我们就令 mid =(l + r + 1)/2,相当于上取整,此时就不会因为r取特殊值r = l + 1而陷入死循环了。

而对于第一个模板,如果左边界l更新为l = mid + 1,是不会出现这样的困扰的。因此,大家可以熟记这两个二分模板,基本上可以解决99%以上的二分问题,再也不会被二分的边界取值所困扰了。

什么时候用模板1?什么时候用模板2?

假设初始时我们的二分区间为[l,r],每次二分缩小区间时,如果左边界l要更新为 l = mid,此时我们就要使用模板2,让 mid = (l + r + 1)/ 2,否则while会陷入死循环。如果左边界l更新为l = mid + 1,此时我们就使用模板1,让mid = (l + r)/2。因此,模板1和模板2本质上是根据代码来区分的,而不是应用场景。如果写完之后发现是l = mid,那么在计算mid时需要加上1,否则如果写完之后发现是l = mid + 1,那么在计算mid时不能加1。

为什么模板要取while( l < r),而不是while( l <= r)?

本质上取l < r 和 l <= r是没有任何区别的,只是习惯问题,如果取l <= r,只需要修改对应的更新区间即可。

while循环结束条件是l >= r,但为什么二分结束时我们优先取r而不是l?

二分的while循环的结束条件是l >= r,所以在循环结束时l有可能会大于r,此时就可能导致越界,二分问题我们优先取r。

二分查找的实现细节:

二分查找时,首先要确定我们要查找的边界值,保证每次二分缩小区间时,边界值始终包含在内。

注意看下面的每张图,最后的答案就是红色箭头指出的位置,也是我们二分的边界值。如果不清楚每次二分时,区间是如何更新的,可以画出和下面类似的图,每次更新区间时,要保证边值始终包含在内,这样关于左右边界的更新就会一目了然。

第一次查找target起始位置:

二分的范围,l = 0, r = nums.size() - 1,我们去二分查找>= target的最左边界begin。

当nums[mid] >= target时,往左半区域找,r = mid。

Java和C++如何在排序数组中查找数字出现的次数

当nums[mid] < target时, 往右半区域找,l = mid + 1。

Java和C++如何在排序数组中查找数字出现的次数

如果nums[r] != target,说明数组中不存在目标值 target,返回 0。否则我们就找到了第一个>=target的位置begin。

第二次查找target结束位置:

二分的范围,l = 0, r = nums.size() - 1,我们去二分查找<= target的最右边界end。

当nums[mid] <= target时,往右半区域找,l = mid。

Java和C++如何在排序数组中查找数字出现的次数

当nums[mid] > target时, 往左半区域找,r = mid - 1。

Java和C++如何在排序数组中查找数字出现的次数

找到了最后一个<= target的位置begin,返回end - begin + 1即可。

时间复杂度分析: 两次二分查找的时间复杂度为 O ( l o g n ) O(logn)O(logn)。

空间复杂度分析: 没有使用额外的数组,因此空间复杂度为O ( 1 ) O(1)O(1)。

3、c++代码

class Solution {public:    int search(vector<int>& nums, int target) {        if(!nums.size()) return  0;        int l = 0, r = nums.size() - 1;        while(l < r)       //查找target的开始位置        {            int mid = (l + r) / 2;            if(nums[mid] >= target) r = mid;            else l = mid + 1;        }        if(nums[r] != target) return 0 ;  //查找失败        int begin = r;     //记录开始位置        l = 0, r = nums.size() - 1;        while(l < r)       //查找tatget的结束位置        {            int mid = (l + r + 1) / 2;            if(nums[mid] <= target) l = mid;            else r = mid - 1;        }        int end = r;       //记录结束位置              return end - begin + 1;    } };

4、java代码

class Solution {    public int search(int[] nums, int target) {        if(nums.length == 0) return  0;        int l = 0, r = nums.length - 1;        while(l < r)       //查找target的开始位置        {            int mid = (l + r) / 2;            if(nums[mid] >= target) r = mid;            else l = mid + 1;        }        if(nums[r] != target) return 0 ;  //查找失败        int begin = r;     //记录开始位置        l = 0; r = nums.length - 1;        while(l < r)       //查找tatget的结束位置        {            int mid = (l + r + 1) / 2;            if(nums[mid] <= target) l = mid;            else r = mid - 1;        }        int end = r;       //记录结束位置              return end - begin + 1;      }}

到此,关于“Java和C++如何在排序数组中查找数字出现的次数”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Java和C++如何在排序数组中查找数字出现的次数

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Java和C++如何在排序数组中查找数字出现的次数

这篇文章主要介绍“Java和C++如何在排序数组中查找数字出现的次数”,在日常操作中,相信很多人在Java和C++如何在排序数组中查找数字出现的次数问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java和C+
2023-06-21

如何用C#找出数组中只出现了一次的数字

数组从字面上理解就是存放一组数,下面这篇文章主要给大家介绍了关于如何用C#找出数组中只出现了一次的数字,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
2022-12-08

python如何查找列表中出现次数最多的数字

小编给大家分享一下python如何查找列表中出现次数最多的数字,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!查找列表中出现次数最多的数字[1, 2, 3, 4,
2023-06-26

java如何找出数组中的不重复数字

找出数组中不重复的一个数字,题目大致是这样的:int[] a = { 1, 2, 3, 4, 3, 2, 1 };在线视频教程推荐:java在线学习解决办法是:public static int getNoRepeat() {int[] a = { 1, 2,
java如何找出数组中的不重复数字
2018-07-23

Python中如何找出序列中出现次数最多的元素

Python中如何找出序列中出现次数最多的元素,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。前言:有一个元素序列,想知道在序列中出现次数最多的元素是什么?col
2023-06-02

Java如何查找指定字符在字符串中的最后一次出现

本文提供了在Java中查找指定字符在字符串中最后一次出现的方法,包括lastIndexOf()、lastChars()、endsWith()和indexOf()方法。lastIndexOf()返回字符的最后一次出现索引,其他方法根据特定情况提供替代方案。理解这些方法对于在Java字符串处理中有效查找字符至关重要。
Java如何查找指定字符在字符串中的最后一次出现
2024-04-02

java中如何实现数组的冒泡排序

冒泡排序依次比较两个相邻的元素,如果前者大于后者就交换位置,每一趟排序之后就会把这趟中的最大值放在最后一位,重复上诉过程,直到没有在需要比较的元素为止。java相关免费学习视频:java学习视频示例如下:public class Bubble_Sort {pu
java中如何实现数组的冒泡排序
2019-09-09

Java如何计算子串在字符串中出现的次数

Java中计算子字符串在字符串中出现的次数有多种方法,包括使用indexOf()、lastIndexOf()、matches()、ApacheCommonsLang3和正则表达式。性能考虑因字符串长度和子字符串长度而异。示例演示了如何使用indexOf()计算“foo”在“barfoobazfoofoo”中的出现次数,结果为3。
Java如何计算子串在字符串中出现的次数
2024-04-02

C语言如何计算子串在字符串中出现的次数

C语言计算字符串中子串出现次数有三种方法:使用strstr()函数,遍历字符串寻找子串,效率较低。结合strstr()和strncmp()函数,提高了效率。采用Boyer-Moore算法,适用于大量字符串处理,效率最高。选择合适的方法取决于字符串长度和出现次数,短字符串和少量出现时,方法1或2即可,大字符串和大批量出现时,Boyer-Moore算法更优。
C语言如何计算子串在字符串中出现的次数
2024-04-02

C++如何实现在有序数组中查找元素的第一个和最后一个位置

这篇文章主要讲解了“C++如何实现在有序数组中查找元素的第一个和最后一个位置”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++如何实现在有序数组中查找元素的第一个和最后一个位置”吧!Fin
2023-06-20

Java如何查找字符串在另一个字符串中第一次出现的位置

本文详细介绍了在Java中查找字符串在另一字符串中首次出现的位置的方法。indexOf()函数、lastIndexOf()函数、正则表达式、contains()函数和循环比较等方法均可用于实现此功能。为提高搜索性能,本文还探讨了Rabin-Karp算法、KMP算法和Trie树等高性能搜索技术。
Java如何查找字符串在另一个字符串中第一次出现的位置
2024-04-02

Java如何查找字符串在另一个字符串中最后一次出现的位置

Java中使用lastIndexOf()方法查找字符串在另一个字符串中最后一次出现的位置,该方法返回子字符串的起始索引,如果未找到则返回-1。该方法区分大小写,也可以使用lastIndexOf(charch)查找单个字符的最后出现位置。
Java如何查找字符串在另一个字符串中最后一次出现的位置
2024-04-02

Python如何计算子串在字符串中出现的次数

Python中计算子串在字符串中出现次数的方法有三种:count()方法:直接获取子串出现次数,速度最快。find()方法:循环查找子串,较慢。re.findall()方法:返回所有匹配子串,可用于复杂模式匹配。count()方法是大多数情况的最佳选择。对于复杂模式匹配,可以使用re.findall()方法。
Python如何计算子串在字符串中出现的次数
2024-04-02

PHP如何计算子串在字符串中出现的次数

本文介绍了在PHP中计算子串在字符串中出现的四种方法:strpos():适用于短字符串和少量子串匹配。preg_match_all():高效,但需要正则表达式。substr_count():专用于计算子串出现次数,简单高效。str_replace():高效,但可能会修改字符串。根据具体需求,选择最合适的方法。
PHP如何计算子串在字符串中出现的次数
2024-04-02

C语言如何查找字符串在另一个字符串中第一次出现的位置

在C语言中,可以通过strstr()、strchr()或自定义函数使用指针来查找字符串在另一个字符串中的第一次出现位置。strstr()用于查找子串,strchr()用于查找字符,而自定义函数提供更灵活的控制。另外,strcmp()、strncmp()和strstrn()函数也可用于某些特定需求。选择方法取决于字符串长度、性能和应用要求。
C语言如何查找字符串在另一个字符串中第一次出现的位置
2024-04-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录