线段树(算法思想+模板+例题)

news2024/9/29 11:32:14

基础思想 : 

介绍

线段树可以用来维护区间信息(区间和 , 区间最值 , 区间GCD等) ,可以在log n的时间内执行区间修改 和 区间查询 ;

1 . 叶子结点的特点是左右结点相等 , 存储元素本身;

2 . 非叶子结点存储的是区间内的统计值 , 

建树

建树的过程相当于dfs : 

#define lc p<<1 ;
#define rc P<<|1 ;
#define N 5e5 + 10 ;
int n , w[N] ;

struct node{
	int l , r , sum , add ;
}tr[N*4];

// 建树 
void build(int p , int l , int r){
	// 对 [l,r] 区间建立线段树 ,当前根的编号为 p 
	tr[p] = {l , r , w[l]} ;
	if(l==r) return ; // 是叶子结点就返回
	int m = l + ((r-l)>>1);
	build(lc,l,m) ;
	build(rc,m+1,r); 
	tr[p].sum = tr[lc].sum + tr[rc].sum ;
}

为什么要开N*4的存储空间 : 

点修改 : 

// 点修改
void update(int p,int x , int k){ // 点修改 : 对点x加上k  : O(logn) 
	// 从根节点出发,递归找到叶子节点,然后从下往上更新其祖先结点 
	if(tr[p].l==x && tr[p].r==x){ // 叶子结点 
		tr[p].sum += k ;
		return ;
	}
	int m = tr[p].l + ((tr[p].r-tr[p].l)>>1) ;//非叶子结点则裂开 
	if(x<=m) udate(lx,x,k) ;
	if(x>m) update(rc,x,k) ;
	tr[p].sum = tr[lc].sum + tr[rc].sum ; 
} 

区间查询 : 

// 区间查询 
int query(int p,int x,int y){ // 查询区间[x,y]的和 : O(logn)
	if(x<=tr[p].l&&tr[p].r<=y){
		return tr[p].sum ;
	}
	int m = tr[p].l + ((tr[p].r-tr[p].l)>>1) ;
	int sum = 0 ;
	if(x<=m) sum += query(lc,x,y);//与左区间重叠 
	if(y>m) sum += query(rc,x,y);//与右区间重叠 
	return sum ;
}

如果带上懒标记 : 

// 区间查询 
int query(int p,int l,int r){ // 查询区间[x,y]的和 : O(logn)
	if(l<=tr[p].l&&tr[p].r<=r){
		return tr[p].sum ;
	}
	pushdown(p) ;
	int m = tr[p].l + tr[p].r >> 1 ;
	int sum = 0 ;
	if(l<=m) sum += query(lc,l,r);//与左区间重叠 
	if(r>m) sum += query(rc,l,r);//与右区间重叠 
	return sum ;
}

区间修改 : 

// 区间修改
void pushup(int p){// 向上更新 
	tr[p].sum=tr[lc].sum+tr[rc].sum; 
} 
void pushdown(int p){ // 向下更新 
	if(tr[p].add){
		tr[lc].sum+=tr[p].add*(tr[lc].r-tr[lc].l+1),
		tr[rc].sum+=tr[p].add*(tr[rc].r-tr[rc].l+1),
		tr[lc].add+=tr[p].add,
		tr[rc].add+=tr[p].add,
		tr[p].add=0;
	}
}
void updateQJ(int p,int x,int y,int k){//对[x,y]全部+k 
	// 先走到叶子节点,将每个叶子节点+k,然后返回修改祖先的结点 
	// 做懒惰标记,当区间[x,y]完全覆盖[a,b],先修改区间的sum值,再打上一个懒标记,然后立即返回
	// 等下次需要是,再下传懒标记,这样就可以将每次修改和查询的时间都控制在O(logn); 
	if(x<=tr[p].l&&tr[p].r<=y){//覆盖则修改
		tr[p].sum += (tr[p].r-tr[p].l+1)*k;
		tr[p].add+=k;
		return ; 
	}
	int m = tr[p].l + ((tr[p].r-tr[p].l)>>1) ;
	pushdown(p);
	if(x<=m) update(lc,x,y,k) ;
	if(y>m) update(rc,x,y,k) ;
	pushup(p);
}

例题 : 

【模板】线段树 1 - 洛谷

#include<bits/stdc++.h>
using namespace std;
#define endl '\n'
#define int long long 
const int N = 1e5 + 10 ;
#define lc p<<1 
#define rc p<<1|1 
int w[N] ;

struct Tree{
	int l , r , sum , add ;
}tr[N*4];

// 区间修改
void pushup(int p){// 向上更新 
	tr[p].sum=tr[lc].sum + tr[rc].sum; 
} 

void pushdown(int p){ // 向下更新 
	if(tr[p].add>0){
		tr[lc].sum+=tr[p].add*(tr[lc].r-tr[lc].l+1);
		tr[rc].sum+=tr[p].add*(tr[rc].r-tr[rc].l+1);
		tr[lc].add+=tr[p].add;
		tr[rc].add+=tr[p].add;
		tr[p].add=0;
	}
}

// 建树 
void build(int p , int l , int r){
	// 对 [l,r] 区间建立线段树 ,当前根的编号为 p 
	tr[p] = {l , r , w[l],0} ;
	if(l==r) return ; // 是叶子结点就返回
	int m = l + r >> 1 ;
	build(lc,l,m) ;
	build(rc,m+1,r); 
	pushup(p) ;
}

void updateQJ(int p,int l,int r,int k){//对[l,r]全部+k 
	// 先走到叶子节点,将每个叶子节点+k,然后返回修改祖先的结点 
	// 做懒惰标记,当区间[l,r]完全覆盖[a,b],先修改区间的sum值,再打上一个懒标记,然后立即返回
	// 等下次需要是,再下传懒标记,这样就可以将每次修改和查询的时间都控制在O(logn); 
	if(l<=tr[p].l&&tr[p].r<=r){//覆盖则修改
		tr[p].sum += (tr[p].r-tr[p].l+1)*k;
		tr[p].add+=k;
		return ; 
	}
	int m = tr[p].l + tr[p].r >> 1 ;
	pushdown(p);
	if(l<=m) updateQJ(lc,l,r,k) ;
	if(r>m) updateQJ(rc,l,r,k) ;
	pushup(p);
}

// 区间查询 
int query(int p,int l,int r){ // 查询区间[x,y]的和 : O(logn)
	if(l<=tr[p].l&&tr[p].r<=r){
		return tr[p].sum ;
	}
	pushdown(p) ;
	int m = tr[p].l + tr[p].r >> 1 ;
	int sum = 0 ;
	if(l<=m) sum += query(lc,l,r);//与左区间重叠 
	if(r>m) sum += query(rc,l,r);//与右区间重叠 
	return sum ;
}

// 点修改
void update(int p,int x , int k){ // 点修改 : 对点x加上k  : O(logn) 
	// 从根节点出发,递归找到叶子节点,然后从下往上更新其祖先结点 
	if(tr[p].l==x && tr[p].r==x){ // 叶子结点 
		tr[p].sum += k ;
		return ;
	}
	int m = tr[p].l + ((tr[p].r-tr[p].l)>>1) ;//非叶子结点则裂开 
	if(x<=m) update(lc,x,k) ;
	if(x>m) update(rc,x,k) ;
	tr[p].sum = tr[lc].sum + tr[rc].sum ; 
} 


signed main(){
	int n , m ; cin >> n >> m ;
	for(int i=1;i<=n;i++) cin >> w[i] ;
	build(1,1,n);
	for(int i=1;i<=m;i++){
		int op ; cin >> op ;
		if(op==1){
			int x , y , k ; cin >> x >> y >> k ;
			updateQJ(1,x,y,k) ;
		}else{
			int x , y ; cin >> x >> y ;
			int ans = query(1,x,y);
			cout << ans << endl ;
		}
	}
}


参考 : 

线段树 - OI Wiki

C02【模板】线段树+懒标记 Luogu P3372 线段树 1_哔哩哔哩_bilibili

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

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

相关文章

力扣|两数相加|链表

给你两个 非空 的链表&#xff0c;表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的&#xff0c;并且每个节点只能存储 一位 数字。 请你将两个数相加&#xff0c;并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外&#xff0c;这两个数都不会以 0 …

用户多部门切换部门,MySQL根据多个部门id递归获取所有上级(祖级)、获取部门的全路径(全结构名称)

背景 之前做过的项目&#xff0c;都是一个用户就一个部门的&#xff0c;现在碰到个一个用户在多个部门的需求&#xff0c;而且需要可以切换不同部门查看不同数据。 就比如说一个大公司下面有多个子公司&#xff0c;每个子公司有好多部门、子部门等等&#xff0c;然后有部分用…

JDK下载配置

一、JDK的作用 Java开发环境&#xff1a;JDK提供了完整的Java开发环境&#xff0c;包含编译器&#xff08;javac&#xff09;、解释器&#xff08;java&#xff09;、打包工具&#xff08;jar&#xff09;、文档生成工具&#xff08;javadoc&#xff09;等一系列工具&#xff0…

人工智能(Educoder)-- 搜索技术 -- 盲目式搜索

第1关&#xff1a;盲目搜索之宽度优先搜索算法 任务描述 本关任务&#xff1a;给定迷宫地图以及在迷宫中的起始位置&#xff0c;利用宽度优先搜索算法求解走出迷宫的最短路径长度&#xff0c;走出迷宫意味着达到迷宫地图的边界&#xff08;所有位置下标0开始&#xff09;。 …

基于python+vue超市在线销售系统的设计与实现flask-django-php-nodejs

根据此问题&#xff0c;研发一套超市在线销售系统&#xff0c;既能够大大提高信息的检索、变更与维护的工作效率&#xff0c;也能够方便信息系统的管理运用&#xff0c;从而减少信息管理成本&#xff0c;提高效率。 该超市在线销售系统采用B/S架构、并采用python语言以及django…

鸿蒙一次开发,多端部署(十二)资源使用

在页面开发过程中&#xff0c;经常需要用到颜色、字体、间距、图片等资源&#xff0c;在不同的设备或配置中&#xff0c;这些资源的值可能不同。有两种方式处理&#xff1a; 应用资源&#xff1a;借助资源文件能力&#xff0c;开发者在应用中自定义资源&#xff0c;自行管理这些…

I/O多路复用:select/poll/epoll

最基本的 Socket 模型 要想客户端和服务器能在网络中通信&#xff0c;那必须得使用 Socket 编程&#xff0c;它是进程间通信里比较特别的方式&#xff0c;特别之处在于它是可以跨主机间通信。 Socket 的中文名叫作插口&#xff0c;咋一看还挺迷惑的。事实上&#xff0c;双方要…

两直线交点算法 C

求两直线交点算法 有中间交点 则CD在AB异侧 A B A C A B A D \nobreak AB \times AC \newline AB \times AD ABACABAD 异号 叉乘后相乘小于零 等于零的几种情况 A B C与AB共线 D与AB共线 求交点&#xff0c;可由面积比例用叉乘计算 C E C D S A B C S A B C D . \frac…

解决前端跨域问题

前端跨域问题 该问题是由于前端的服务路径或端口和后台的不一致所导致的 Springboot跨域设置 import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.cors.CorsConfiguration; …

「媒体邀约」选择媒体公关公司邀约媒体有哪些优势-51媒体网

传媒如春雨&#xff0c;润物细无声&#xff0c;大家好&#xff0c;我是51媒体网胡老师。 选择媒体公关公司邀约媒体具有以下优势&#xff1a; 丰富的媒体资源&#xff1a;媒体公关公司通常与各大主流媒体、行业媒体、网络媒体等有着长期合作关系&#xff0c;拥有丰富的媒体资源…

ChatGPT论文指南|揭秘8大ChatGPT提示词研究技巧提升写作效率【建议收藏】

点击下方▼▼▼▼链接直达AIPaperPass &#xff01; AIPaperPass - AI论文写作指导平台 公众号原文▼▼▼▼&#xff1a; ChatGPT论文指南|揭秘8大ChatGPT提示词研究技巧提升写作效率【建议收藏】 目录 1.写作方法 2.方法设计 3.研究结果 4.讨论写作 5.总结结论 6.书…

Install Docker

Docker Desktop 直接安装 Docker Desktop Docker Desktop includes the Docker daemon (dockerd), the Docker client (docker), Docker Compose, Docker Content Trust, Kubernetes, and Credential Helper. Linux下安装Docker CE 参考官方文档 参见阿里云的文档 # step 1…

ClickHouse--11--物化视图

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 1.物化视图什么是物化视图? 1.1 普通视图1.2 物化视图1.3 优缺点1.4 基本语法1.5 在生产环境中创建物化视图1.6 AggregatingMergeTree 表引擎3.1 概念3.2 Aggregat…

WorkPlus一站式IM即时通讯解决方案,提升企业沟通效率与协作能力

在企业内部沟通与协作中&#xff0c;高效的即时通讯是实现团队协作与工作效率的重要保障。而WorkPlus以其稳定可靠的性能和全面的功能&#xff0c;为企业提供一站式的IM即时通讯解决方案&#xff0c;助力企业提升沟通效率与协作能力。IM即时通讯在企业中的重要性不言而喻。作为…

【索引失效】MySQL索引失效场景

1、对索引使用左或者左右模糊匹配 当我们使用左或者左右模糊匹配的时候&#xff0c;也就是 like %xx 或者 like %xx% 这两种方式都会造成索引失效。 比如下面的 like 语句&#xff0c;查询 name 后缀为「林」的用户&#xff0c;执行计划中的 typeALL 就代表了全表扫描&#xff…

解决Animate.css动画效果无法在浏览器运行问题

背景 在开发官方网站的时候&#xff0c;临时更换了电脑&#xff0c;发现原本正常的动画效果突然不动了。 经过 chrome、Microsoft Edge都无法运行。 Animate.css | A cross-browser library of CSS animations. 问题排查 通过审查元素后发现类名是注入并且生效的。 验证 然…

【Linux】vim配置及安装方法

注 安装方法在文章最后 配置文件的位置 在目录 /etc/ 下面&#xff0c;有个名为vimrc的文件&#xff0c;这是系统中公共的vim配置文件&#xff0c;对所有用户都有效。而在每个用户的主目录下&#xff0c;都可以自己建立私有的配置文件&#xff0c;命名为“.vimrc”。例如&…

小目标检测篇 | YOLOv8改进之增加小目标检测层(四头检测机制)

前言:Hello大家好,我是小哥谈。小目标检测是计算机视觉领域中的一个研究方向,旨在从图像或视频中准确地检测和定位尺寸较小的目标物体。相比于常规目标检测任务,小目标检测更具挑战性,因为小目标通常具有低分辨率、低对比度和模糊等特点,容易被背景干扰或遮挡。为了解决小…

Windows复现SiamCAR代码遇到的报错与解决方法

一、环境基础 Windows10以上 已装Anaconda 支持GPU 已经gitclone:https://github.com/HonglinChu/SiamTrackers 二、遇到的报错 1. No module named pycocotools._mask 方案一&#xff1a;加载非常慢 conda install -c conda-forge pycocotools 方…

Yocto学习笔记1-下载与首次编译

Yocto学习笔记1-下载与首次编译 1、基础环境介绍2、注意点3、安装依赖3.1 yocto常规系统构建所需依赖库&#xff08;较全&#xff09;3.2 龙芯适配时的最小依赖库&#xff08;最小&#xff09; 4、下载4.1 通过git克隆4.2 查看所有远程分支4.3 签出一个长期支持的稳定版本4.4 查…