AcWing算法基础课-789数的范围-Java题解

news2024/11/13 14:46:57

heweilai-bolg-title-image-of-the-article

大家好,我是何未来,本篇文章给大家讲解《AcWing算法基础课》789 题——数的范围。本文详细解析了一个基于二分查找的算法题,题目要求在有序数组中查找特定元素的首次和最后一次出现的位置。通过使用两个二分查找函数,程序能够高效地处理大量查询,时间复杂度为 O(n log n),空间复杂度为 O(n)。文章从题目描述、算法思路、具体实现步骤到Java代码实现,全面展示了如何解决这一问题。

文章目录

  • ❓题目描述
  • 💡算法思路
  • ✅Java代码
  • 🔗参考

❓题目描述

给定一个按照升序排列的长度为 n 的整数数组,以及 q 个查询。

对于每个查询,返回一个元素 k 的起始位置和终止位置(位置从 0 开始计数)。

如果数组中不存在该元素,则返回 -1 -1

输入格式

第一行包含整数 n 和 q,表示数组长度和询问个数。

第二行包含 n 个整数(均在 1∼10000 范围内),表示完整数组。

接下来 q 行,每行包含一个整数 k,表示一个询问元素。

输出格式

共 q 行,每行包含两个整数,表示所求元素的起始位置和终止位置。

如果数组中不存在该元素,则返回 -1 -1

数据范围

1≤n≤100000
1≤q≤10000
1≤k≤10000

输入样例:

6 3
1 2 2 3 3 4
3
4
5

输出样例:

3 4
5 5
-1 -1

💡算法思路

  1. 使用二分查找算法findFristGreatOrEqual(int l, int r, int x)查找数组中元素 k 的开始位置
  2. 如果找到了 k 的开始位置,就返回元素 k 的开始位置,并使用findFirstLessOrEqual(int l, int r, int x)查找 k 的结束位置
  3. 如果没有找到 k 的开始位置,那么也不存在 k 的结束位置。直接返回-1 -1

具体实现步骤:

  1. 初始化输入流

    • 程序首先创建一个 StreamTokenizer 对象,用于从标准输入流中读取数据。这个对象通过 BufferedReaderInputStreamReader 来处理输入。
  2. 读取数组大小和查询次数

    • 程序调用 nextInt() 方法两次,分别读取数组的大小 n 和查询次数 q
  3. 读取数组元素

    • 程序使用一个循环,调用 nextInt() 方法 n 次,将输入的整数依次存储到数组 nums 中。
  4. 处理每个查询

    • 程序使用一个循环,处理 q 次查询。每次查询时,程序调用 nextInt() 方法读取一个整数 k,表示要查找的值。
  5. 查找首次出现的位置

    • 程序调用 findFristGreatOrEqual(0, n - 1, k) 方法,使用二分查找算法在数组中查找第一个大于或等于 k 的元素的位置。
    • 如果找到的位置上的元素等于 k,则说明 k 存在于数组中。
  6. 查找最后一次出现的位置

    • 如果 k 存在于数组中,程序调用 findFirstLessOrEqual(0, n - 1, k) 方法,使用二分查找算法在数组中查找第一个小于或等于 k 的元素的位置。
    • 这个位置即为 k 在数组中最后一次出现的位置。
  7. 输出结果

    • 如果 k 存在于数组中,程序输出 k 的首次和最后一次出现的位置。
    • 如果 k 不存在于数组中,程序输出 -1 -1
  8. 二分查找函数的实现

    •   findFristGreatOrEqual(int l, int r, int x)
      
      • 这个函数使用二分查找算法,在数组的 [l, r] 范围内查找第一个大于或等于 x 的元素的位置。
      • 通过不断缩小查找范围,最终找到目标位置。
    •   findFirstLessOrEqual(int l, int r, int x)
      
      • 这个函数使用二分查找算法,在数组的 [l, r] 范围内查找第一个小于或等于 x 的元素的位置。
      • 通过不断缩小查找范围,最终找到目标位置。

时间复杂度:O(n log n)
空间复杂度:O(n)

✅Java代码

import java.*;

public class Aw789 {

	// 创建一个StreamTokenizer对象,用于读取输入流
	static StreamTokenizer in = new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));

	// 读取下一个整数
	static int nextInt() throws IOException {
		in.nextToken();
		return (int) in.nval;
	}

	static int n, q, k;
	// 定义一个数组来存储输入的数字,数组大小为100010
	static int[] nums = new int[100000 + 10];

	public static void main(String[] args) throws IOException {
		// 读取数组的大小n和查询次数q
		n = nextInt();
		q = nextInt();
		// 读取n个整数并存储到数组nums中
		for (int i = 0; i < n; i++) {
			nums[i] = nextInt();
		}
		// 处理q次查询
		while (q-- > 0) {
			// 读取查询的数字k
			k = nextInt();
			// 查找第一个大于或等于k的元素的位置
			int left = findFristGreatOrEqual(0, n - 1, k);
			// 如果找到的元素等于k
			if (nums[left] == k) {
				// 输出第一个等于k的元素的位置和最后一个等于k的元素的位置
				System.out.println(left + " " + findFirstLessOrEqual(0, n - 1, k));
			} else {
				// 如果没有找到等于k的元素,输出-1 -1
				System.out.println("-1 -1");
			}
		}
	}

	// 二分查找第一个大于或等于x的元素的位置
	static int findFristGreatOrEqual(int l, int r, int x) {
		while (l < r) {
			int mid = l + r >> 1; // 计算中间位置
			if (nums[mid] >= x) {
				r = mid; // 如果中间元素大于或等于x,缩小右边界
			} else {
				l = mid + 1; // 否则,缩小左边界
			}
		}
		return l; // 返回找到的位置
	}

	// 二分查找第一个小于或等于x的元素的位置
	static int findFirstLessOrEqual(int l, int r, int x) {
		while (l < r) {
			int mid = (l + r + 1) >> 1; // 计算中间位置
			if (nums[mid] <= x) {
				l = mid; // 如果中间元素小于或等于x,缩小左边界
			} else {
				r = mid - 1; // 否则,缩小右边界
			}
		}
		return l; // 返回找到的位置
	}

}

🔗参考

  • https://www.acwing.com/problem/content/791/

作者:程序员何未来-heweilai.com


🔍推荐阅读

  1. AcWing算法基础课-788逆序对的数量-Java题解
  2. AcWing算法基础课-787归并排序-Java题解
  3. 博客赚钱全攻略:从新手到专家的变现之路

欢迎关注我的博客:@程序员何未来,持续为你输出有价值的技术文章~
你们的点赞👍 收藏⭐ 留言🗨️ 关注✅
是我持续创作,输出优质内容的最大动力!谢谢!

文章关键词:算法,计算机算法,算法题解,算法竞赛,Java,数据结构,AcWing算法基础课

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数据结构(Day13)

一、学习内容 内存空间划分 1、一个进程启动后&#xff0c;计算机会给该进程分配4G的虚拟内存 2、其中0G-3G是用户空间【程序员写代码操作部分】【应用层】 3、3G-4G是内核空间【与底层驱动有关】 4、所有进程共享3G-4G的内核空间&#xff0c;每个进程独立拥有0G-3G的用户空间 …

【C++】深入理解作用域和命名空间:从基础到进阶详解

&#x1f984;个人主页:小米里的大麦-CSDN博客 &#x1f38f;所属专栏:C_小米里的大麦的博客-CSDN博客 &#x1f381;代码托管:C: 探索C编程精髓&#xff0c;打造高效代码仓库 (gitee.com) ⚙️操作环境:Visual Studio 2022 目录 一、前言 二、域的概念 1. 类域 2. 命名空间…

Redis——常用数据类型string

目录 常用数据结构&#xff08;类型&#xff09;Redis单线程模型Reids为啥效率这么高&#xff1f;速度这么快&#xff1f;&#xff08;参照于其他数据库&#xff09; stringsetgetMSET 和 MGETSETNX&#xff0c;SETEX&#xff0c;PSETEXincr&#xff0c;incrby&#xff0c;decr…

sshj使用代理连接服务器

之前我是用jsch连接服务器的&#xff0c;但是没办法使用私钥连接&#xff0c;搜了一下似乎是不支持新版的SSH-rsa&#xff0c;并且jsch很久没更新了&#xff0c;java - "com.jcraft.jsch.JSchException: Auth fail" with working passwords - Stack Overflow 没办法…

mybatis的基本使用与配置

注释很详细&#xff0c;直接上代码 项目结构 源码 UserMapper package com.amoorzheyu.mapper;import com.amoorzheyu.pojo.User; import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Select;import java.util.List;Mapper //在运行时生成代…

从数据仓库到数据中台再到数据飞轮:金融行业的数据技术进化史

前言​ 大家好&#xff0c;我是一名大数据开发工程师&#xff0c;在金融行业深耕多年&#xff0c;其实数据技术的演进不仅是技术层面的革新&#xff0c;更是业务模式与决策方式的深刻变革。从最开始的数据仓库兴起&#xff0c;到数据中台的普及&#xff0c;再到数据飞轮的出现…

MFEA/D-DRA--基于分解和动态资源分配的多目标多任务优化

MFEA/D-DRA–基于分解和动态资源分配的多目标多任务优化 title&#xff1a; A Multiobjective multifactorial optimization algorithm based on decomposition and dynamic resource allocation strategy author&#xff1a; Shuangshuang Yao, Zhiming Dong, Xianpeng Wang…

跨界融合,GIS如何赋能游戏商业——以《黑神话:悟空》为例

在数字化时代&#xff0c;地理信息系统&#xff08;GIS&#xff09;技术正以其独特的空间分析和可视化能力&#xff0c;为游戏产业带来革命性的变革。《黑神话&#xff1a;悟空》作为中国首款3A级别的动作角色扮演游戏&#xff0c;不仅在游戏设计和技术上取得了突破&#xff0c…

金融行业中如何利用数据中台的数据来有效的驱动业务决策呢?

前言​ 在金融行业中&#xff0c;利用数据中台的数据来有效驱动业务决策是一个复杂而关键的过程。其实我们的核心就是帮助金融机构最大化数据中台的价值&#xff0c;并推动业务决策的科学性和准确性。本文我从技术的角度来剖析一下这一过程。​ 什么是数据中台&#xff1f;​…

Git常用指令大全详解

Git常用指令大全详解 Git&#xff0c;作为目前最流行的分布式版本控制系统&#xff0c;其强大的功能和灵活性为开发者提供了极大的便利。无论是个人项目还是团队协作&#xff0c;Git都扮演着不可或缺的角色。本文将详细总结Git的常用指令&#xff0c;帮助大家更好地掌握这一工…

PHP:强大的Web开发语言

PHP&#xff1a;强大的Web开发语言 一、PHP 简介及优势 PHP 的基本概念 PHP&#xff08;PHP: Hypertext Preprocessor&#xff09;即 “超文本预处理器”&#xff0c;是一种通用开源脚本语言&#xff0c;最初由 Rasmus Lerdorf 于 1994 年创建。它可以在服务器上执行&#xf…

题目:单调栈

1、关于栈的概述 栈是一种数据结构&#xff0c;遵循“后进先出”&#xff08;LIFO, Last In, First Out&#xff09;的原则。这意味着最后被插入栈中的元素会最先被移除。可以把它想象成一个垒盘子的情况&#xff0c;新的盘子总是放在最上面&#xff0c;而最上面的盘子会最先被…

Matlab:科学计算与工程应用的强大利器

Matlab&#xff1a;强大的科学计算工具 一、Matlab 简介与重要性 Matlab 作为一款强大的科学计算软件&#xff0c;在工程、科学、数学等多个领域都有着广泛的应用及至关重要的地位。 在工程计算领域&#xff0c;它涵盖了众多方面。例如&#xff0c;线性代数与矩阵运算中&…

CODESYS资源使用表

1、CODESYS标准化编程之输入输出映射请参考下面文章链接: CODESYS标准化编程之输入输出映射-CSDN博客文章浏览阅读78次。在介绍输入输出映射之前大家需要了解开关量防抖滤波功能块,相关链接如下:开关量防抖滤波器(梯形图和SCL源代码)_开关量输入滤波程序-CSDN博客文章浏览阅…

禹神:一小时彻底搞懂跨域解决方案

1. 浏览器的同源策略 2. 跨域会受到哪些限制 4. CORS 解决 Ajax 跨域问题 exposedHeaders 不加这个&#xff0c;js拿不到这个响应头(浏览器控制台network中能看见&#xff0c;但是js拿不到) 5. JSONP 解决跨域问题 JSOP只能解决get请求 服务端代码 客户端代码 服务端代码升…

Gartner发布报告揭秘微软数据安全功能和许可

制定数据安全计划以增强合规性并降低数据风险仍然是安全和风险管理领导者关注的问题。这项研究阐明了 Microsoft 的数据安全许可结构&#xff0c;并确定了围绕 Purview 构建数据安全计划的关键要素。 主要发现 客户对微软数据安全的询问表明&#xff0c;安全和风险管理 (SRM) 领…

transformer模型进行英译汉,汉译英

上面是在测试集上的表现 下面是在训练集上的表现

全面掌控大模型:MaxKB与Ollama的高效本地部署策略

随着大模型的广泛应用&#xff0c;越来越多的开发者希望能够在本地运行这些模型&#xff0c;既提高数据隐私性&#xff0c;又避免依赖云端服务。本文将详细介绍如何在本地使用 Ollama 进行大模型部署&#xff0c;以及如何通过 MaxKB 导入本地知识库并进行交互操作。为了使该过程…

在线包装盒型生成工具,各种异型包装盒型,PDF导出方便

1、templatemaker.nl Passepartout ✂ Templatemaker ︎https://www.templatemaker.nl/en/passepartout/这是一个荷兰设计师建的一个在线盒型自动生成工具&#xff0c;包含各类新奇盒型&#xff0c;大家可以一起去观摩一下。 网站首页顶部各种盒型展示&#xff0c;大家根据需…

【CTF MISC】XCTF GFSJ1088 [中等] QR1 Writeup(图像处理+QR Code识别)

[中等] QR1 一张空白的图片&#xff1f; 解法 一张空白图片。 用 Photoshop 打开&#xff0c;放大&#xff0c;发现很多小黑点。 将图片复制到新文档&#xff0c;用魔棒工具选择白色部分。 Ctrl Shift i 反选。编辑&#xff0c;描边&#xff0c;黑色&#xff0c;10px&#…