数据结构 - 线段树

news2025/1/17 4:09:50

1. 预制值:

  • 构建的数组为,nums:【2, 5, 1, 4, 3】
  • 区间和问题,假设求区间 [1,3] 的和

2. 建树

2.1 构建线段树数组

int[] segT = new int[4*n]

(为什么数组大小是4*n???评论区欢迎留言)
在这里插入图片描述

tips:长方格中的left、right,分别代表该节点所求得的区间和。例如,left:0,right:4。代表nums中索引0到4的和=2+5+1+4+3=15。

在这里插入图片描述

  • 核心代码
public void build(int index, int left, int right) {

		// 2.1.1的工作,递归条件,不可再分区间
		if (left == right) {
			segT[index] = nums[left];
			return;
		}
		
		// 2.1.1的工作,区间多次二分
		int mid = (right - left) / 2 + left;
		
		// 左子树根节点的索引
		int leftIndex = index * 2;
		// 右子树根节点的索引
		int rightIndex= index * 2 + 1;
		
		// 构建左、右子树
		build(leftIndex, left, mid);
		build(rightIndex, mid+1, right);
		
		// 2.1.2的工作,求根节点的和
		// tips:只有当left==right返回后,才会走这一步
		segT[index] = segT[leftIndex] + segT[rightIndex];
		
	}

3. 更新

逻辑其实和建树一样,在线段树中找到修改节点的对应索引,然后修改其值,然后再依次修改根节点的值即可。
在这里插入图片描述

  • 核心代码
public void update(int index, int start, int end, int updateIndex, int value) {

		// 找到叶子节点,直接修改值(不可再分区间了)
		if (start == end) {
			nums[updateIndex] = value;
			segT[index] = value;
			return;
		} 
		
		int mid = (start + end) / 2;
		int leftIndex = index * 2;
		int rightIndex = index * 2 + 1;
		
		// 修改的节点再左子树还是右子树
		if (updateIndex >= start && updateIndex <= mid) {
			update(leftIndex, start, mid, updateIndex, value);
		} else {
			update(rightIndex, mid + 1, end, updateIndex, value);
		}
		segT[index] = segT[leftIndex] + segT[rightIndex];
		
	}

4. 查询

查询的逻辑会稍微复杂一点,因为有三个因素:

4.1 查询的范围全落在左子树

在这里插入图片描述

4.2 查询的范围全落在右子树

在这里插入图片描述

4.3 查询的范围既在左子树、又在右子树

在这里插入图片描述

  • 核心代码
public int querySum(int index, int start, int end, int left, int right) {
		
		// 查询的索引无效(不是线段树的有意义范围)
		if (left > end || right < start) {
			return 0;
		}
		// 查询的范围包含该子树的范围,直接返回即可(见下图【包含子树】)
		else if (left <= start && right >= end) {
			return st[index].sum;
		}
		
		// 求的左右子树的索引
		int mid = (start + end) / 2;
		int leftIndex = index * 2;
		int rightIndex = index * 2 + 1;
		
		// 左、右子树求得的值
		int leftSum = querySum(leftIndex, start, mid, left, right);
		int rightSum = querySum(rightIndex, mid + 1, end, left, right);
		
		return leftSum + rightSum;
	}

在这里插入图片描述
图:包含子树

5. 线段树求区间问题模板(最大值、最小值、和)



public class SegmentTree {
	
	class Node {
		int left;
		int right;
		int max;
		int min;
		int sum;
		
		Node() {}
		
		Node (int left, int right) {
			this.left = left;
			this.right = right;
			this.max = Integer.MIN_VALUE;
			this.min = Integer.MAX_VALUE;
			this.sum = 0;
		}
	}
	
	
	
	int n;
	Node[] st;
	int[] nums;
	
	public void build(int index, int left, int right) {
		
		if (left == right) {
			st[index].sum = nums[left];
			st[index].max = nums[left];
			st[index].min = nums[left];
			return;
		}
		
		int mid = (right - left) / 2 + left;
		
		int leftIndex = index * 2;
		int rightIndex= index * 2 + 1;
		
		build(leftIndex, left, mid);
		build(rightIndex, mid+1, right);
		
		st[index].max = Math.max(st[leftIndex].max, st[rightIndex].max);
		st[index].min = Math.min(st[leftIndex].min, st[rightIndex].min);
		st[index].sum = st[leftIndex].sum + st[rightIndex].sum;
		
	}
	
	
	
	
	public void init(int N, int[] arr) {
		n = N;
		st = new Node[4 * n + 1];
		for (int i = 1; i <= 4 * n; i++) {
			st[i] = new Node();
		}
		nums = arr;
	}
	
	public int querySum(int left, int right) {
		
		return querySum(1, 0, n-1, left, right);
	}
	
	
	public int querySum(int index, int start, int end, int left, int right) {
		
		if (left > end || right < start) {
			return 0;
		} else if (left <= start && right >= end) {
			return st[index].sum;
		}
		

		int mid = (start + end) / 2;
		int leftIndex = index * 2;
		int rightIndex = index * 2 + 1;
		
		int leftSum = querySum(leftIndex, start, mid, left, right);
		int rightSum = querySum(rightIndex, mid + 1, end, left, right);
		
		return leftSum + rightSum;
	}
	
	
	public int queryMax(int left, int right) {
		return queryMax(1, 0, n-1, left, right);
	}
	
	public int queryMax(int index, int start, int end, int left, int right) {
		
		if (left > end || right < start) {
			return 0;
		} else if (left <= start && right >= end) {
			return st[index].sum;
		}
		

		int mid = (start + end) / 2;
		int leftIndex = index * 2;
		int rightIndex = index * 2 + 1;
		
		int leftMax = queryMax(leftIndex, start, mid, left, right);
		int rightMax = queryMax(rightIndex, mid + 1, end, left, right);
		
		return Math.max(leftMax, rightMax);
	}
	
	public int queryMin(int left, int right) {
		return queryMin(1, 0, n-1, left, right);
	}
	
	public int queryMin(int index, int start, int end, int left, int right) {
		if (left > end || right < start) {
			return Integer.MAX_VALUE;
		} else if (left <= start && right >= end) {
			return st[index].sum;
		}
		

		int mid = (start + end) / 2;
		int leftIndex = index * 2;
		int rightIndex = index * 2 + 1;
		
		int leftMin = queryMin(leftIndex, start, mid, left, right);
		int rightMin = queryMin(rightIndex, mid + 1, end, left, right);
		
		return Math.min(leftMin, rightMin);
	}
	
	
	public void update(int index, int start, int end, int updateIndex, int value) {
		if (start == end) {
			nums[updateIndex] = value;
			st[index].sum = value;
			st[index].max = value;
			st[index].min = value;
			return;
		} 
		
		int mid = (start + end) / 2;
		int leftIndex = index * 2;
		int rightIndex = index * 2 + 1;
		
		if (updateIndex >= start && updateIndex <= mid) {
			update(leftIndex, start, mid, updateIndex, value);
		} else {
			update(rightIndex, mid + 1, end, updateIndex, value);
		}
		st[index].max = Math.max(st[leftIndex].max, st[index].max);
		st[index].min = Math.min(st[leftIndex].min, st[index].min);
		st[index].sum = st[leftIndex].sum + st[index].sum;
		
	}
	
	
	public void update(int index, int value) {
		nums[index] = value;
		update(1, 0, n - 1, index, value);
	}
	
}

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

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

相关文章

Vivado Tri-MAC IP的例化配置(三速以太网IP)

目录 1 Tri-MAC IP使用RGMII接口的例化配置1.1 Data Rate1.2 interface配置1.3 Shared Logic配置1.4 Features 2 配置完成IP例化视图 1 Tri-MAC IP使用RGMII接口的例化配置 在网络设计中&#xff0c;使用的IP核一般为三速以太网IP核&#xff0c;使用时在大多数场景下为配置为三…

【计算机图形学】实验一 DDA算法、Bresenham算法

&#x1f57a;作者&#xff1a; 主页 我的专栏C语言从0到1探秘C数据结构从0到1探秘Linux &#x1f618;欢迎关注&#xff1a;&#x1f44d;点赞&#x1f64c;收藏✍️留言 &#x1f3c7;码字不易&#xff0c;你的&#x1f44d;点赞&#x1f64c;收藏❤️关注对我真的很重要&…

【Git】05 分离头指针

文章目录 一、分离头指针二、创建分支三、比较commit内容四、总结 一、分离头指针 正常情况下&#xff0c;在通过git checkout命令切换分支时&#xff0c;在命令后面跟着的是分支名&#xff08;例如master、temp等&#xff09;或分支名对应commit的哈希值。 非正常情况下&…

python_蓝桥杯刷题记录_笔记_全AC代码_入门3

前言 记录我的解法以及笔记思路&#xff0c;谢谢观看。 题单目录 1.P2141 [NOIP2014 普及组] 珠心算测验 2.P1567 统计天数 3.P1055 [NOIP2008 普及组] ISBN 号码 4.P1200 [USACO1.1] 你的飞碟在这儿 Your Ride Is Here 5.P1308 [NOIP2011 普及组] 统计单词数 6.P1047 […

Kubernetes的有状态应用示例:用PV部署WordPress和MySQL

文章目录 环境PVC和PV创建PVC和PV 创建kustomization.yaml添加secret生成器 为MySQL和WordPress添加资源配置部署和验证清理参考 环境 RHEL 9.3Docker Community 24.0.7minikube v1.32.0 PVC和PV PersistentVolume&#xff08;PV&#xff09;是在集群里由管理员手动provisio…

Python爬虫urllib详解

前言 学习爬虫&#xff0c;最初的操作便是模拟浏览器向服务器发出请求&#xff0c;那么我们需要从哪个地方做起呢&#xff1f;请求需要我们自己来构造吗&#xff1f;需要关心请求这个数据结构的实现吗&#xff1f;需要了解 HTTP、TCP、IP 层的网络传输通信吗&#xff1f;需要知…

vue3 之 组合式API—computed

computed计算属性函数 计算属性基本思想和Vue2的完全一致&#xff0c;组合式API下的计算属性只是修改了写法 核心步骤&#xff1a; 导入computed函数执行函数 在回调参数中return基于响应式数据做计算的值&#xff0c;用变量接收 vue <script setup> // 1.导入compute…

【数据分享】1929-2023年全球站点的逐年降雪深度数据(Shp\Excel\免费获取)

气象数据是在各项研究中都经常使用的数据&#xff0c;气象指标包括气温、风速、降水、能见度等指标&#xff0c;说到气象数据&#xff0c;最详细的气象数据是具体到气象监测站点的数据&#xff01; 之前我们分享过1929-2023年全球气象站点的逐年平均气温数据、逐年最高气温数据…

TypeScript实战系列之强力爆破泛型的困扰

目录 介绍开始如何理解泛型语法泛型约束泛型默认值练习后续 介绍 泛型在typescript 中使用频率相当高&#xff0c;也给了初学者相当大的阻碍。希望这一篇文章&#xff0c;能帮助你们爆破它。 开始 下面通过模拟实现一个简易版本的axios来引入泛型的使用 // axios.ts type M…

10秒部署好 Serverless Web,我只告诉你一个人

谁还不知道Serverless&#xff1f; 上个月亚马逊云科技 re:Invent大会第一个演讲主题就是它。 每一个新技术概念的引入都伴随着生产力的提高&#xff0c;Serverless指的是一种构建和运行不需要服务器管理的应用程序的概念。 一个很有用的技术&#xff0c;这篇文章带你入门。…

alibabacloud学习笔记05(小滴课堂)

高并发下的微服务存在的问题 高并发下的微服务容错方案 介绍什么是分布式系统的流量防卫兵Sentinel 微服务引入Sentinel和控制台搭建 每个服务都加上这个依赖。 启动方式&#xff1a; 讲解AliababCloud微服务整合Sentinel限流配置实操 我们在order和video模块都加上。 分别启动…

网站接入QQ登录

开发者认证&#xff0c;创建应用&#xff1a; &#xff08;1&#xff09;先在 QQ 互联 QQ互联管理中心 进行个人开发者认证&#xff0c;并审核 &#xff08;2&#xff09;创建【网站应用】&#xff0c;填写正确信息&#xff0c;获得对应的 appid 与 appkey&#xff0c;并审核…

Redis——SpringBoot整合Redis实战

1、基本配置 1.1、引入依赖 首先&#xff0c;建立Maven项目&#xff0c;在Maven项目中引入pom.xml文件&#xff1a; <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> &l…

【HarmonyOS应用开发】后台提醒(十六)

简述 随着生活节奏的加快&#xff0c;我们有时会忘记一些重要的事情或日子&#xff0c;所以提醒功能必不可少。应用可能需要在指定的时刻&#xff0c;向用户发送一些业务提醒通知。例如购物类应用&#xff0c;希望在指定时间点提醒用户有优惠活动。为满足此类业务诉求&#xf…

机器学习 | 解析聚类算法在数据检测中的应用

目录 初识聚类算法 聚类算法实现流程 模型评估 算法优化 特征降维 探究用户对物品类别的喜好细分(实操) 初识聚类算法 聚类算法是一种无监督学习方法&#xff0c;用于将数据集中的对象按照相似性分组。它旨在发现数据中的内在结构和模式&#xff0c;将具有相似特征的数据…

Photoshop CS6 下载安装教程,保姆级教程,小白也能轻松搞的,附安装包

前言 Adobe Photoshop CS6强大的照片拍摄和突破性的新功能&#xff0c;用于复杂的图形、选择、逼真的绘画和装饰智能。创建惊人的高动态范围(HDR)图像。用逼真的笔触和混合的颜色绘画。消除噪音&#xff0c;添加种子&#xff0c;并绘制一个国家最先进的摄影设备的草图。凭借原…

【HarmonyOS应用开发】ArkUI 开发框架-进阶篇-Video组件的使用(十)

一、Video组件的使用 1、概述 在手机、平板或是智慧屏这些终端设备上&#xff0c;媒体功能可以算作是我们最常用的场景之一。无论是实现音频的播放、录制、采集&#xff0c;还是视频的播放、切换、循环&#xff0c;亦或是相机的预览、拍照等功能&#xff0c;媒体组件都是必不可…

MacBook有必要装清理软件吗?CleanMyMac的一些主要特点

MacBook是苹果公司的一款高端笔记本电脑&#xff0c;但是&#xff0c;随着使用时间的增长&#xff0c;MacBook也会出现一些问题&#xff0c;比如运行缓慢、卡顿、垃圾文件堆积、磁盘空间不足等。这些问题不仅影响了用户的使用体验&#xff0c;也可能对MacBook的寿命和安全性造成…

Shell脚本⑧免交互

目录 一.Here Document 1.定义 2.变量 &#xff08;1&#xff09;变量替换成实际值 &#xff08;2&#xff09;整行内容作为变量并输出结果 &#xff08;3&#xff09;多行注释 &#xff08;4&#xff09;自动划分磁盘免交互 二.Expect 1.定义 2.安装 3.免交互操作 …

django线上教育学习平台大数据分析系统python

随着互联网技术不断地发展&#xff0c;网络与大数据成为了人们生活的一部分&#xff0c;而线上教育平台大数据分析作为网上应用的一个全新的体现&#xff0c;由于其特有的便捷性&#xff0c;已经被人们所接受。目前主流的线上教育平台大数据分析服务不仅不明确并且管理盈利较低…