2023-07-26力扣每日一题-区间翻转线段树
链接:
2569. 更新数组后处理求和查询
题意:
给两个等长数组nums1和nums2,三个操作:
操作1:将nums1的[l,r]翻转(0变1,1变0)
操作2:将nums2[any]变成nums2[any]+nums1[any]*p,p由操作给出,any表示数组里的每一位
操作3:查询nums2的和
解:
由于每次更新nums2的时候,不需要考虑nums2[any]本身的值(基于nums2[any]增减,但增减多少不取决于nums2[any]),而且每次查询都查询整个nums2,所以只要维护一个nums2的和sum,不用考虑修改nums2
这样实际上我们只需要一颗区间翻转型线段树,节点设计:
struct Node
{int l,r,val;//左端点,右端点,值bool lazy;//区间翻转使用bool型懒标记即可Node():l(-1),r(-1),val(0),lazy(false) {}; //构造函数
};
然后就是设计线段树,4N空间:
struct SegTree
{vector<Node>SegNodes;//存储线段树节点vector<int>& nums;//设计了nums1的引用,方便类内访问nums1SegTree(vector<int> &nums1): nums(nums1)//绑定应用{SegNodes.resize(4*nums1.size());//4倍空间}
};
(下面都是类内函数)构建函数设计,使用递归求和,这边注意运算符优先级(n<<1)|1 和 n<<1|1一样,但是(n<<1)+2 和 n<<1+2不一样:
int Build(int n,int l,int r){//cout<<"Build"<<n<<ends<<l<<ends<<r<<endl; SegNodes[n].l=l;//设定这个节点 应有 的左端点SegNodes[n].r=r;//设定这个节点 应有 的右端点if(l==r) return SegNodes[n].val=nums[l];//当左端点等于右端点时,没有子节点,直接从num中取值else//有子节点{int mid=(l+r)/2;//区间中点SegNodes[n].val+=Build((n<<1)|1,l,mid);//左子节点SegNodes[n].val+=Build((n<<1)+2,mid+1,r);//右子节点return SegNodes[n].val;//返回求值结果}}
懒标记传递函数,每次使用该节点时且存在懒标记时才进行传递,对一段区间反复修改的时候不需要对最底层修改:
void pushdown(int n)//向下传递懒标记 {//cout<<"push_down"<<n<<endl; if(SegNodes[n].lazy)//存在懒标记{SegNodes[n].lazy=false;//移除当前懒标记 SegNodes[(n<<1)|1].lazy=!SegNodes[(n<<1)|1].lazy;//更新子节点标记SegNodes[(n<<1)|1].val=SegNodes[(n<<1)|1].r-SegNodes[(n<<1)|1].l+1-SegNodes[(n<<1)|1].val;SegNodes[(n<<1)+2].lazy=!SegNodes[(n<<1)+2].lazy;//更新子节点标记SegNodes[(n<<1)+2].val=SegNodes[(n<<1)+2].r-SegNodes[(n<<1)+2].l+1-SegNodes[(n<<1)+2].val;}}
区间修改函数,如果当前修改段包含整个区间,那么直接整个区间打上懒标记;如果不是包含整个区间,则对这个区间的左右子节点进行判断修改:
void change(int n,int l,int r){//cout<<"change"<<n<<ends<<l<<ends<<r<<endl; if(SegNodes[n].l>=l && SegNodes[n].r<=r)//包含整个区间 {SegNodes[n].lazy=!SegNodes[n].lazy;SegNodes[n].val=SegNodes[n].r-SegNodes[n].l+1-SegNodes[n].val;return;//直接修改+返回}pushdown(n);//看看该节点有没有需要传递的标记if(SegNodes[(n<<1)|1].r>=l) change((n<<1)|1,l,r);//查看左节点if(SegNodes[(n<<1)+2].l<=r) change((n<<1)+2,l,r);//查看右节点SegNodes[n].val=SegNodes[(n<<1)|1].val+SegNodes[(n<<1)+2].val;//更新父节点值}
区间查询函数,同样如果包含整个区间,直接返回值,否则查询左右子节点:
int query(int n,int l,int r){if(SegNodes[n].l>=l && SegNodes[n].r<=r) return SegNodes[n].val;if(SegNodes[n].r<l || SegNodes[n].l>r) return 0;pushdown(n);int ret=0;if(SegNodes[(n<<1)|1].r>=l) ret+=query((n<<1)|1,l,r);if(SegNodes[(n<<1)+2].l<=r) ret+=query((n<<1)+2,l,r);return ret;}
实际代码:
#include<bits/stdc++.h>
using namespace std;
struct Node
{int l,r,val;bool lazy;Node():l(-1),r(-1),val(0),lazy(false) {};
};
struct SegTree
{vector<Node>SegNodes;vector<int>& nums;SegTree(vector<int> &nums1): nums(nums1){SegNodes.resize(4*nums1.size());}void check(){cout<<"checkIng"<<endl;for(auto &SegNode:SegNodes){cout<<SegNode.l<<" "<<SegNode.r<<" "<<SegNode.val<<endl;cout<<"lazy:"<<SegNode.lazy<<endl; }cout<<"checkEnd"<<endl;}int Build(int n,int l,int r){//cout<<"Build"<<n<<ends<<l<<ends<<r<<endl; SegNodes[n].l=l;SegNodes[n].r=r;if(l==r) return SegNodes[n].val=nums[l];else{int mid=(l+r)/2;SegNodes[n].val+=Build((n<<1)|1,l,mid);SegNodes[n].val+=Build((n<<1)+2,mid+1,r);return SegNodes[n].val;}}void pushdown(int n)//向下传递懒标记 {//cout<<"push_down"<<n<<endl; if(SegNodes[n].lazy)//存在懒标记{SegNodes[n].lazy=false;//移除当前懒标记 SegNodes[(n<<1)|1].lazy=!SegNodes[(n<<1)|1].lazy;//更新子节点标记SegNodes[(n<<1)|1].val=SegNodes[(n<<1)|1].r-SegNodes[(n<<1)|1].l+1-SegNodes[(n<<1)|1].val;SegNodes[(n<<1)+2].lazy=!SegNodes[(n<<1)+2].lazy;//更新子节点标记SegNodes[(n<<1)+2].val=SegNodes[(n<<1)+2].r-SegNodes[(n<<1)+2].l+1-SegNodes[(n<<1)+2].val;}}void change(int n,int l,int r){//cout<<"change"<<n<<ends<<l<<ends<<r<<endl; if(SegNodes[n].l>=l && SegNodes[n].r<=r)//包含整个区间 {SegNodes[n].lazy=!SegNodes[n].lazy;SegNodes[n].val=SegNodes[n].r-SegNodes[n].l+1-SegNodes[n].val;return;}pushdown(n);if(SegNodes[(n<<1)|1].r>=l) change((n<<1)|1,l,r);if(SegNodes[(n<<1)+2].l<=r) change((n<<1)+2,l,r);SegNodes[n].val=SegNodes[(n<<1)|1].val+SegNodes[(n<<1)+2].val;}int query(int n,int l,int r){if(SegNodes[n].l>=l && SegNodes[n].r<=r) return SegNodes[n].val;if(SegNodes[n].r<l || SegNodes[n].l>r) return 0;pushdown(n);int ret=0;if(SegNodes[(n<<1)|1].r>=l) ret+=query((n<<1)|1,l,r);if(SegNodes[(n<<1)+2].l<=r) ret+=query((n<<1)+2,l,r);return ret;}
};
vector<long long> handleQuery(vector<int>& nums1, vector<int>& nums2, vector<vector<int>>& queries)
{int lg=nums1.size();long long sum=0;vector<long long>ans;for(const auto &num2:nums2) sum+=num2;SegTree st(nums1);st.Build(0,0,lg-1);for(const auto &querie:queries){//st.check();switch(querie[0]){case 1:st.change(0,querie[1],querie[2]);break;case 2:sum+=(long long)st.query(0,0,lg-1)*querie[1];break;case 3:ans.push_back(sum);break;}}return ans;
}
int main()
{vector<vector<int>> queries;vector<int>nums1,nums2;int n1,n2,temp,temp0;cout<<"input n1 and n2 as size(nums1) and size(nums2)"<<endl; cin>>n1>>n2;while(n1--){cin>>temp;nums1.push_back(temp);}while(n2--){cin>>temp;nums2.push_back(temp);}cout<<"input n1 as size(queries)"<<endl; cin>>n1;while(n1--){cout<<"input mode key1 key2"<<endl;cin>>n2>>temp>>temp0;queries.push_back({n2,temp,temp0});}vector<long long>ans=handleQuery(nums1,nums2,queries);for(auto a:ans) cout<<a<<ends;cout<<endl;return 0;
}
限制:
1 <= nums1.length,nums2.length <= 105nums1.length = nums2.length1 <= queries.length <= 105queries[i].length = 30 <= l <= r <= nums1.length - 10 <= p <= 1060 <= nums1[i] <= 10 <= nums2[i] <= 109
相关文章:
2023-07-26力扣每日一题-区间翻转线段树
链接: 2569. 更新数组后处理求和查询 题意: 给两个等长数组nums1和nums2,三个操作: 操作1:将nums1的[l,r]翻转(0变1,1变0) 操作2:将nums2[any]变成nums2[any]nums1[any]*p&…...
Java设计模式之 -- 桥接模式
什么是桥接模式 桥接模式是一种结构型设计模式,也被称为“Handle/Body”。这种设计模式主要用于将抽象部分与它的实现部分分离,使它们可以独立地变化。这种方式有助于减少系统中的耦合性,增加了扩展性。 主要解决什么问题 桥接模式主要解决…...
【node.js】02-path模块
目录 1. path.join() 2. path.basename() 3. path.extname() 1. path.join() 使用 path.join() 方法,可以把多个路径片段拼接为完整的路径字符串,语法格式如下: path.join([...paths]) 例子: const path require(path)co…...
攻防世界-Reverse-re1
题目描述:菜鸡开始学习逆向工程,首先是最简单的题目 下载附件,执行程序,如下界面 1. 思路分析 没啥说的,既然题目都说是一道简单的逆向题,那么直接使用ida逆向即可,看逆向出的结果是否能写入到…...
AES加密的基本常识和封装类
AES加密的基本常识和封装类 AES(Advanced Encryption Standard)是一种对称密钥加密算法,被广泛用于保护敏感数据的安全性。它是一种块加密算法,意味着它将明文数据分成固定大小的块,并使用相同的密钥对每个块进行独立…...
elasticsearch使用记录
参考文章:https://elasticsearch-py.readthedocs.io/en/v8.8.2/ 参考文章:https://cuiqingcai.com/6214.html 参考文章:https://www.cnblogs.com/cupleo/p/13953890.html elasticsearch版本:8.8.2(软件包发行版) python版本&#…...
UNI-APP_横屏切换竖屏出现样式混乱问题
app从竖屏页面1进入竖屏页面2,再进入横屏,再返回,再返回从新回到竖屏页面1,再次进入竖屏页面2,发现竖屏页面2的所有图片字体都被放大了。再返回竖屏1,再进入竖屏2,一切又恢复正常。 解决跳转横…...
数据可视化(3)
1.饼状图 #饼状图 #pie(x,labels,colors,labeldistance,autopct,startangle,radius,center,textprops) #x,每一块饼状图的比例 #labels:每一块饼形图外侧显示的文字说明 #labeldistance:标记的绘制位置,相对于半径的比例…...
AI面试官:MD5、DES、RSA、AES加密
AI面试官:MD5、DES、RSA、AES加密 文章目录 AI面试官:MD5、DES、RSA、AES加密1. 什么是MD5加密?它在实际应用中有哪些场景?2. DES加密是什么?它在现实中的应用场景有哪些?3. 问题:RSA加密是什么…...
Shell脚本学习-$$特殊变量
$$特殊变量: 获取脚本执行的进程号(PID)。 [rootvm1 scripts]# cat test_pid.sh echo $$ > /tmp/a.pid sleep 300代码说明: 1)获取$$值,也就是当前脚本进程的PID值,重定向到/tmp/a.pid文件…...
vscode中python插件过新导致无法正常debug问题解决安装vscode以前版本python插件教程
您需要从.vsix文件安装它。您可以在此处找到它们。 下载所需.vsix版本的文件。您可能需要单击assets才能看到它们。 然后打开 VSCode,转到extensions-> 单击三个点 ->install from vsix并选择您的文件。 重启以后,就可以正常debug了!...
chrome macos编译
下载工具包 git clone https://chromium.googlesource.com/chromium/tools/depot_tools/gitpwd export PATH"$PATH:/Users/lichengjun/Downloads/chrome_build/depot_tools" mkdir chromium cd chromium 如果想快的话直接: fetch --nohooks --no-history chromium (…...
Linux环境下Elasticsearch相关软件安装
Linux环境下Elasticsearch相关软件安装 本文将介绍在linux(Centos7)环境下安装Elasticsearch相关的软件。 1、安装Elasticsearch 1.1 Elasticsearch下载 首先去Elasticsearch官网下载相应版本的安装包,下载之后传输到linux服务器上。 官网地址:http…...
【趟坑记录】d3.zoom()的正确使用姿势 @d3.v7
【趟坑记录】d3.zoom()的正确使用姿势 d3.v7 文章目录 【趟坑记录】d3.zoom()的正确使用姿势 d3.v7问题重现原因分析解决方案放缩平移写法特殊修改transform函数的写法 总结 在开发一个D3应用的时候遇到了一个 zoom相关的问题,记录解决思路与方案 问题重现 最近在…...
基于 Docker + Nginx + Gitlab-runner 实现前端自动化部署流程
本篇会用到Docker,Gitlab-runner等相关工具,如果对其不是特别了解,可以参考下相关文档: GitLab RunnerDocker 快速入门CI/CD:持续集成/持续部署 在早期部署前端项目时,我们通常会通过ftp把前端代码直接传…...
make/makefile的使用
make/makefile 文章目录 make/makefile初步认识makefile的工作流程依赖关系和依赖方法make的使用 总结 make是一个命令,是一个解释makefile中指令的命令工具,makefile是一个文件,当前目录下的文件,两者搭配使用,完成项…...
Flutter中Navigator 跳转传参数和反向传参数
初始化路由 MaterialApp(routes: <String, WidgetBuilder>{"/Second": (BuildContext context){return Second("");}}, 跳转传参数 String va await Navigator.of(context).push(MaterialPageRoute(builder: (content) {return Second( demo); },…...
kettle开发-Day40-AI分流之case/switch
前言: 前面我们讲到了很多关于数据流的AI方面的介绍,包括自定义组件和算力提升这块的,今天我们来学习一个关于kettle数据分流处理非常重要的组件Switch / Case 。当我们的数据来源于类似日志、csv文件等半结构化数据时,我们需要在…...
MySQL下载与安装
MySQL下载与安装 一、下载 地址:https://dev.mysql.com/downloads/mysql/ 当前最新是8.0版本,我选择上一个最新的mysql-5.7.24-winx64.zip 二、安装 MySQL安装文件分两种 .msi和.zip ,.msi需要安装 zip格式是自己解压,解压缩之后…...
c++基础2
文件操作 程序运行时产生的数据属于临时数据,程序一旦运行结束都会被释放 通过文件可以将数据持久化 c中对文件操作需要包含 文件类型分为两种 文本文件:文件以ASCII码形式存储在计算机中二进制文件:文件以文本的二进制存储在计算机中&a…...
Objective-C常用命名规范总结
【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名(Class Name)2.协议名(Protocol Name)3.方法名(Method Name)4.属性名(Property Name)5.局部变量/实例变量(Local / Instance Variables&…...
Linux-07 ubuntu 的 chrome 启动不了
文章目录 问题原因解决步骤一、卸载旧版chrome二、重新安装chorme三、启动不了,报错如下四、启动不了,解决如下 总结 问题原因 在应用中可以看到chrome,但是打不开(说明:原来的ubuntu系统出问题了,这个是备用的硬盘&a…...
OpenLayers 分屏对比(地图联动)
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 地图分屏对比在WebGIS开发中是很常见的功能,和卷帘图层不一样的是,分屏对比是在各个地图中添加相同或者不同的图层进行对比查看。…...
《C++ 模板》
目录 函数模板 类模板 非类型模板参数 模板特化 函数模板特化 类模板的特化 模板,就像一个模具,里面可以将不同类型的材料做成一个形状,其分为函数模板和类模板。 函数模板 函数模板可以简化函数重载的代码。格式:templa…...
RabbitMQ入门4.1.0版本(基于java、SpringBoot操作)
RabbitMQ 一、RabbitMQ概述 RabbitMQ RabbitMQ最初由LShift和CohesiveFT于2007年开发,后来由Pivotal Software Inc.(现为VMware子公司)接管。RabbitMQ 是一个开源的消息代理和队列服务器,用 Erlang 语言编写。广泛应用于各种分布…...
解决:Android studio 编译后报错\app\src\main\cpp\CMakeLists.txt‘ to exist
现象: android studio报错: [CXX1409] D:\GitLab\xxxxx\app.cxx\Debug\3f3w4y1i\arm64-v8a\android_gradle_build.json : expected buildFiles file ‘D:\GitLab\xxxxx\app\src\main\cpp\CMakeLists.txt’ to exist 解决: 不要动CMakeLists.…...
redis和redission的区别
Redis 和 Redisson 是两个密切相关但又本质不同的技术,它们扮演着完全不同的角色: Redis: 内存数据库/数据结构存储 本质: 它是一个开源的、高性能的、基于内存的 键值存储数据库。它也可以将数据持久化到磁盘。 核心功能: 提供丰…...
规则与人性的天平——由高考迟到事件引发的思考
当那位身着校服的考生在考场关闭1分钟后狂奔而至,他涨红的脸上写满绝望。铁门内秒针划过的弧度,成为改变人生的残酷抛物线。家长声嘶力竭的哀求与考务人员机械的"这是规定",构成当代中国教育最尖锐的隐喻。 一、刚性规则的必要性 …...
文件上传漏洞防御全攻略
要全面防范文件上传漏洞,需构建多层防御体系,结合技术验证、存储隔离与权限控制: 🔒 一、基础防护层 前端校验(仅辅助) 通过JavaScript限制文件后缀名(白名单)和大小,提…...
Python学习(8) ----- Python的类与对象
Python 中的类(Class)与对象(Object)是面向对象编程(OOP)的核心。我们可以通过“类是模板,对象是实例”来理解它们的关系。 🧱 一句话理解: 类就像“图纸”,对…...
