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…...
反向工程与模型迁移:打造未来商品详情API的可持续创新体系
在电商行业蓬勃发展的当下,商品详情API作为连接电商平台与开发者、商家及用户的关键纽带,其重要性日益凸显。传统商品详情API主要聚焦于商品基本信息(如名称、价格、库存等)的获取与展示,已难以满足市场对个性化、智能…...
VB.net复制Ntag213卡写入UID
本示例使用的发卡器:https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...
React Native 开发环境搭建(全平台详解)
React Native 开发环境搭建(全平台详解) 在开始使用 React Native 开发移动应用之前,正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南,涵盖 macOS 和 Windows 平台的配置步骤,如何在 Android 和 iOS…...
SCAU期末笔记 - 数据分析与数据挖掘题库解析
这门怎么题库答案不全啊日 来简单学一下子来 一、选择题(可多选) 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘:专注于发现数据中…...
学习STC51单片机31(芯片为STC89C52RCRC)OLED显示屏1
每日一言 生活的美好,总是藏在那些你咬牙坚持的日子里。 硬件:OLED 以后要用到OLED的时候找到这个文件 OLED的设备地址 SSD1306"SSD" 是品牌缩写,"1306" 是产品编号。 驱动 OLED 屏幕的 IIC 总线数据传输格式 示意图 …...
CRMEB 框架中 PHP 上传扩展开发:涵盖本地上传及阿里云 OSS、腾讯云 COS、七牛云
目前已有本地上传、阿里云OSS上传、腾讯云COS上传、七牛云上传扩展 扩展入口文件 文件目录 crmeb\services\upload\Upload.php namespace crmeb\services\upload;use crmeb\basic\BaseManager; use think\facade\Config;/*** Class Upload* package crmeb\services\upload* …...
3-11单元格区域边界定位(End属性)学习笔记
返回一个Range 对象,只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意:它移动的位置必须是相连的有内容的单元格…...
ip子接口配置及删除
配置永久生效的子接口,2个IP 都可以登录你这一台服务器。重启不失效。 永久的 [应用] vi /etc/sysconfig/network-scripts/ifcfg-eth0修改文件内内容 TYPE"Ethernet" BOOTPROTO"none" NAME"eth0" DEVICE"eth0" ONBOOT&q…...
MFC 抛体运动模拟:常见问题解决与界面美化
在 MFC 中开发抛体运动模拟程序时,我们常遇到 轨迹残留、无效刷新、视觉单调、物理逻辑瑕疵 等问题。本文将针对这些痛点,详细解析原因并提供解决方案,同时兼顾界面美化,让模拟效果更专业、更高效。 问题一:历史轨迹与小球残影残留 现象 小球运动后,历史位置的 “残影”…...
苹果AI眼镜:从“工具”到“社交姿态”的范式革命——重新定义AI交互入口的未来机会
在2025年的AI硬件浪潮中,苹果AI眼镜(Apple Glasses)正在引发一场关于“人机交互形态”的深度思考。它并非简单地替代AirPods或Apple Watch,而是开辟了一个全新的、日常可接受的AI入口。其核心价值不在于功能的堆叠,而在于如何通过形态设计打破社交壁垒,成为用户“全天佩戴…...
