力扣1109. 航班预订统计 差分入门模板题 附线段树解法

2023-10-31 06:04

本文主要是介绍力扣1109. 航班预订统计 差分入门模板题 附线段树解法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原题

本题属于「区间求和」问题中的入门难度。

差分解法:

class Solution {/*本题只涉及「区间修改 + 单点查询」,因此是一道「差分」的模板题。「差分」可以看做是求「前缀和」的逆向过程。对于一个「将区间[L,R]整体增加一个值 V」操作,我们可以对差分数组 C 的影响看成两部分:对 C[L]+=V:由于差分是前缀和的逆向过程,这个操作对于将来的查询而言,带来的影响是对于所有的下标大于等于L的位置都增加了值V;对 C[R+1]-=V:由于我们期望只对[L,R]产生影响,因此需要对下标大于R的位置进行减值操作,从而抵消“影响”。对于最后的构造答案,可看做是对每个下标做“单点查询”操作,只需要对差分数组求前缀和即可:
*/public int[] corpFlightBookings(int[][] bookings, int n) {int pre[]=new int[n+1],ans[]=new int[n]; //pre[]必须声明成n+1 因为C[R+1]-=V时R+1有可能越界了for (int[] booking : bookings) {int firsti=booking[0],lasti=booking[1],seatsi=booking[2];pre[firsti-1]+=seatsi;pre[lasti]-=seatsi;}ans[0]=pre[0];for (int i = 1; i < n; i++) {ans[i]=ans[i-1]+pre[i];}return ans;}
}

线段树可以无脑解决所有的「区间求和」问题,但是很多时候没必要,代码量太大,表现也不是最好的:

class Solution {public int[] corpFlightBookings(int[][] bookings, int n) {for (int[] booking : bookings) {int l=booking[0]-1,r=booking[1]-1,seatsi=booking[2];update(root,0,n,l,r,seatsi);}int ans[] = new int[n];for (int i = 0; i < n; i++) {ans[i]=query(root,0,n,i,i);}return ans;}class Node {Node left, right;int val, add;}private int N = (int) 2e4;//线段树范围大小private Node root = new Node();// 在区间 [start, end] 中更新区间 [l, r] 的值,将区间 [l, r] ➕ val// 如果结点表示为「区间最值」的情况时在更新结点时不需要✖️叶子节点的数量// 下面的pushDown函数里也是 也要根据题目判断是否需要✖️叶子节点的数量// 如果是「点更新」 也不需要✖️叶子节点的数量  因为此时区间是一个点所以一定会更新到叶子节点//start, end一般是用0和N 对应root的范围!public void update(Node node, int start, int end, int l, int r, int val) {if (l <= start && end <= r) {node.val += (end - start + 1) * val;//✖️叶子节点的数量node.add += val;//这里俩个+=如果在点赋值时可以改成= 点累加不行return ;}int mid = (start + end) >> 1;pushDown(node, mid - start + 1, end - mid);if (l <= mid) update(node.left, start, mid, l, r, val);if (r > mid) update(node.right, mid + 1, end, l, r, val);pushUp(node);}// 在区间 [start, end] 中查询区间 [l, r] 的结果, [l ,r] 保持不变public int query(Node node, int start, int end, int l, int r) {if (l <= start && end <= r) return node.val;int mid = (start + end) >> 1, ans = 0;pushDown(node, mid - start + 1, end - mid);if (l <= mid) ans += query(node.left, start, mid, l, r);if (r > mid) ans += query(node.right, mid + 1, end, l, r);return ans;}// 向上更新private void pushUp(Node node) {node.val = node.left.val + node.right.val;}// 推懒惰标记的函数// 如果是「覆盖」的更新操作 则在下推懒惰标记的时候『不需要累加』private void pushDown(Node node, int leftNum, int rightNum) {if (node.left == null) node.left = new Node();if (node.right == null) node.right = new Node();if (node.add == 0) return ;// 当前节点加上标记值✖️该子树所有叶子节点的数量 同上面update函数同步修改node.left.val += node.add * leftNum;node.right.val += node.add * rightNum;//这里俩个+=如果在点赋值时可以改成= 点累加不行// 对区间进行「加减」的更新操作时,下推懒惰标记时需要累加起来,不能直接覆盖node.left.add += node.add;node.right.add += node.add;node.add = 0;}
}

这篇关于力扣1109. 航班预订统计 差分入门模板题 附线段树解法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/313330

相关文章

opencv实现像素统计的示例代码

《opencv实现像素统计的示例代码》本文介绍了OpenCV中统计图像像素信息的常用方法和函数,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 统计像素值的基本信息2. 统计像素值的直方图3. 统计像素值的总和4. 统计非零像素的数量

如何使用 Bash 脚本中的time命令来统计命令执行时间(中英双语)

《如何使用Bash脚本中的time命令来统计命令执行时间(中英双语)》本文介绍了如何在Bash脚本中使用`time`命令来测量命令执行时间,包括`real`、`user`和`sys`三个时间指标,... 使用 Bash 脚本中的 time 命令来统计命令执行时间在日常的开发和运维过程中,性能监控和优化是不

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

hdu1496(用hash思想统计数目)

作为一个刚学hash的孩子,感觉这道题目很不错,灵活的运用的数组的下标。 解题步骤:如果用常规方法解,那么时间复杂度为O(n^4),肯定会超时,然后参考了网上的解题方法,将等式分成两个部分,a*x1^2+b*x2^2和c*x3^2+d*x4^2, 各自作为数组的下标,如果两部分相加为0,则满足等式; 代码如下: #include<iostream>#include<algorithm

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n