2024蓝桥杯省赛保奖突击班-Day2-前缀和、差分、尺取_笔记_练习题解

本文主要是介绍2024蓝桥杯省赛保奖突击班-Day2-前缀和、差分、尺取_笔记_练习题解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

3月25日-课堂笔记

  • 前缀和预处理 O ( n ) \mathcal{O}(n) O(n)
s[1] = a[1];
for(int i = 2; i <= n; ++ i)s[i] = s[i - 1] + a[i];
  • 利用前缀和查询区间和 O ( 1 ) O(1) O(1)
long long calc(int l, int r) {return l == 1 ? s[r] : s[r] - s[l - 1];
}
  • 差分序列的求法
c[1] = a[1];
for(int i = 2; i <= n; ++ i) c[i] = a[i] - a[i - 1];
  • 原序列上区间[l, r]修改相当于差分序列上两个单点修改
c[l] += v;
c[r + 1] -= v;
  • 区间加等差数列对应二次差分序列上常数个单点修改
  • 一般等差数列: …, 0 , a , a + d , a + 2 d , … , a + ( k − 1 ) d , 0 , 0 , … 0, a, a+d, a+2 d, \ldots, a+(k-1) d, 0,0, \ldots 0,a,a+d,a+2d,,a+(k1)d,0,0,
  • 一次差分之后: …, 0 , a , d , d , … , d , − a − ( k − 1 ) d , 0 , … 0, a, d, d, \ldots, d,-a-(k-1) d, 0, \ldots 0,a,d,d,,d,a(k1)d,0,
  • 二次差分之后: …, 0 , a , d − a , 0 , … , 0 , − a − k d , a + ( k − 1 ) d , … 0, a, d-a, 0, \ldots, 0,-a-k d, a+(k-1) d, \ldots 0,a,da,0,,0,akd,a+(k1)d,
  • 尺取法
for(int l = 1, r = 0; r <= n; ++ l) {while(num < m && r < n) ...;if(...) break;...
}
  • 双栈法维护尺取
    插入/删除 -> 插入/合并

练习题解

B3612 求区间和

题目链接:B3612
参考思路

前缀和模板题。

C++参考代码
#include <iostream>
#include <vector>using namespace std;int main() {int n, m;cin >> n;vector<int> a(n), prefix_sum(n + 1, 0);for (int i = 0; i < n; ++i)cin >> a[i];for (int i = 1; i <= n; ++i)prefix_sum[i] = prefix_sum[i - 1] + a[i - 1];cin >> m;while (m--) {int l, r;cin >> l >> r;cout << prefix_sum[r] - prefix_sum[l - 1] << endl;}return 0;
}
Java参考代码
import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int n = scanner.nextInt();int[] a = new int[n];int[] prefixSum = new int[n + 1];for (int i = 0; i < n; i++)a[i] = scanner.nextInt();for (int i = 1; i <= n; i++)prefixSum[i] = prefixSum[i - 1] + a[i - 1];int m = scanner.nextInt();while (m-- > 0) {int l = scanner.nextInt();int r = scanner.nextInt();System.out.println(prefixSum[r] - prefixSum[l - 1]);}}
}
Python参考代码
n = int(input())
a = list(map(int, input().split()))prefix_sum = [0] * (n + 1)
for i in range(1, n + 1):prefix_sum[i] = prefix_sum[i - 1] + a[i - 1]m = int(input())
for _ in range(m):l, r = map(int, input().split())print(prefix_sum[r] - prefix_sum[l - 1])

P2367 语文成绩

题目链接:P2367
参考思路
C++参考代码
#include <iostream>
#include <vector>using namespace std;int main() {int n, p;cin >> n >> p;vector<int> scores(n), diff(n, 0);for (int i = 0; i < n; ++i)cin >> scores[i];while (p--) {int x, y, z;cin >> x >> y >> z;diff[x - 1] += z;if(y < n) diff[y] -= z;}for (int i = 1; i < n; ++i)diff[i] += diff[i - 1];int min_score = scores[0] + diff[0];for (int i = 1; i < n; ++i) {scores[i] += diff[i];min_score = min(min_score, scores[i]);}cout << min_score << endl;return 0;
}
Java参考代码
import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int n = scanner.nextInt();int p = scanner.nextInt();int[] scores = new int[n];int[] diff = new int[n];for (int i = 0; i < n; i++)scores[i] = scanner.nextInt();while (p-- > 0) {int x = scanner.nextInt();int y = scanner.nextInt();int z = scanner.nextInt();diff[x - 1] += z;if (y < n)diff[y] -= z;}int minScore = scores[0] + diff[0];for (int i = 1; i < n; i++) {diff[i] += diff[i - 1];scores[i] += diff[i];minScore = Math.min(minScore, scores[i]);}System.out.println(minScore);}
}
Python参考代码
n, p = map(int, input().split())
scores = list(map(int, input().split()))
diff = [0] * nfor _ in range(p):x, y, z = map(int, input().split())diff[x - 1] += zif y < n:diff[y] -= zmin_score = scores[0] + diff[0]
for i in range(1, n):diff[i] += diff[i - 1]scores[i] += diff[i]min_score = min(min_score, scores[i])print(min_score)

P3406 海底高铁

题目链接:P3406
思路:本题可以使用差分数组和前缀和求出每一段需要经过的次数 再用贪心策略在2种买票方式的最优中选择。
C++参考代码
#include <iostream>
#include <algorithm>using namespace std;
long long c[100005] = {0};
int main() {int n, m;cin >> n >> m;long long sum = 0, ans = 0;int p1 = 0, p2 = 0, a, b, c1;if (m > 0) {cin >> p1;}for (int i = 2; i <= m; i++) {cin >> p2;if (p1 < p2) {c[p1]++;c[p2]--;} else {c[p2]++;c[p1]--;}p1 = p2;}for (int i = 1; i < n; i++) {sum += c[i];cin >> a >> b >> c1;if (sum != 0) {ans += min(a * sum, b * sum + c1);}}if (m <= 1) {ans = 0;}cout << ans;return 0;
}
Java参考代码
import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int n = scanner.nextInt();int m = scanner.nextInt();long[] c = new long[100005];long sum = 0, ans = 0;int p1 = 0, p2 = 0, a, b, c1;if (m > 0) {p1 = scanner.nextInt();}for (int i = 2; i <= m; i++) {p2 = scanner.nextInt();if (p1 < p2) {c[p1]++;c[p2]--;} else {c[p2]++;c[p1]--;}p1 = p2;}for (int i = 1; i < n; i++) {sum += c[i];a = scanner.nextInt();b = scanner.nextInt();c1 = scanner.nextInt();if (sum != 0) {ans += Math.min(a * sum, b * sum + c1);}}if (m <= 1) {ans = 0;}System.out.println(ans);scanner.close();}
}
Python参考代码
n, m = map(int, input().split())
c = [0] * 100005
sum = 0
ans = 0
P = list(map(int,input().split()))
p1 = P[0]
for i in range(1,len(P)):p2 = P[i]if p1 < p2:c[p1] += 1c[p2] -= 1else:c[p2] += 1c[p1] -= 1p1 = p2for i in range(1, n):sum += c[i]a, b, c1 = map(int, input().split())if sum != 0:ans += min(a * sum, b * sum + c1)if m <= 1:ans = 0print(ans)

P4552 IncDec Sequence

题目链接:P4552
参考思路:(这是一个比较困难的题)

题中要使所有数都一样。那么,也就是说,在差分的数组中,除了第一个数字外,其他的数字必须为0
那么我们要做的,就是使除了第一个数字外,其他的数字必须为0。
我们知道差分的公式为 c [ l ] + = v ; c [ r + 1 ] − = v c[l]+=v;c[r+1]-=v c[l]+=v;c[r+1]=v;
那么我们可以得出结论:
最少次数就是在差分序列中的正数相加的值负数相加的绝对值的较大值。
那么,如何解决方法的种数呢?这又是转换法。
思考,差分后的第一个数字的种数是不是就是题目要求的方法数量。
那么要改变差分的第一个数字,是不是以 c [ 1 ] + + , c [ i ] − − 或 c [ 1 ] − − , c [ i ] + + c[1]++,c[i]--或c[1]--,c[i]++ c[1]++,c[i]c[1],c[i]++的方法来改变。
由于要求步数最少,要在差分数组中所有的正数或负数已经和其他数相互抵消完后,才能用 c [ 1 ] c[1] c[1]来勾兑。
那么答案就是正数和负数的绝对值的最大值减去正数和负数的绝对值的最小值。

C++参考代码
#include<bits/stdc++.h>
#define ll long long
using namespace std;
ll n,sum1=0,sum2=0;
int main()
{cin >> n;vector<ll>a(n+1,0);for(int i = 1 ; i <= n ; i++){cin >> a[i];}vector<ll>C(n+1,0);for(int i = 2 ; i <= n ; i++){C[i] = a[i] - a[i - 1];if(C[i] > 0) sum1 += C[i];else sum2 -= C[i];}cout << max(sum1 , sum2) << endl ;cout << abs(sum1 - sum2) + 1 << endl ;return 0;
}
Java参考代码
import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);int n = scanner.nextInt();long[] a = new long[n + 1];  for (int i = 1; i <= n; i++) {a[i] = scanner.nextLong();}long[] C = new long[n + 1];  long sum1 = 0, sum2 = 0;for (int i = 2; i <= n; i++) {C[i] = a[i] - a[i - 1];if (C[i] > 0) sum1 += C[i];else sum2 -= C[i];}System.out.println(Math.max(sum1, sum2));System.out.println(Math.abs(sum1 - sum2) + 1);scanner.close();}
}
Python参考代码
n = int(input())
a = [0] * (n + 1)  
for i in range(1, n + 1):a[i] = int(input())C = [0] * (n + 1)
sum1 = sum2 = 0
for i in range(2, n + 1):C[i] = a[i] - a[i - 1]if C[i] > 0:sum1 += C[i]else:sum2 -= C[i]  print(max(sum1, sum2))
print(abs(sum1 - sum2) + 1)

这篇关于2024蓝桥杯省赛保奖突击班-Day2-前缀和、差分、尺取_笔记_练习题解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

2024年流动式起重机司机证模拟考试题库及流动式起重机司机理论考试试题

题库来源:安全生产模拟考试一点通公众号小程序 2024年流动式起重机司机证模拟考试题库及流动式起重机司机理论考试试题是由安全生产模拟考试一点通提供,流动式起重机司机证模拟考试题库是根据流动式起重机司机最新版教材,流动式起重机司机大纲整理而成(含2024年流动式起重机司机证模拟考试题库及流动式起重机司机理论考试试题参考答案和部分工种参考解析),掌握本资料和学校方法,考试容易。流动式起重机司机考试技

【专题】2024飞行汽车技术全景报告合集PDF分享(附原数据表)

原文链接: https://tecdat.cn/?p=37628 6月16日,小鹏汇天旅航者X2在北京大兴国际机场临空经济区完成首飞,这也是小鹏汇天的产品在京津冀地区进行的首次飞行。小鹏汇天方面还表示,公司准备量产,并计划今年四季度开启预售小鹏汇天分体式飞行汽车,探索分体式飞行汽车城际通勤。阅读原文,获取专题报告合集全文,解锁文末271份飞行汽车相关行业研究报告。 据悉,业内人士对飞行汽车行业

高效录音转文字:2024年四大工具精选!

在快节奏的工作生活中,能够快速将录音转换成文字是一项非常实用的能力。特别是在需要记录会议纪要、讲座内容或者是采访素材的时候,一款优秀的在线录音转文字工具能派上大用场。以下推荐几个好用的录音转文字工具! 365在线转文字 直达链接:https://www.pdf365.cn/ 365在线转文字是一款提供在线录音转文字服务的工具,它以其高效、便捷的特点受到用户的青睐。用户无需下载安装任何软件,只

poj 3159 (spfa差分约束最短路) poj 1201

poj 3159: 题意: 每次给出b比a多不多于c个糖果,求n最多比1多多少个糖果。 解析: 差分约束。 这个博客讲差分约束讲的比较好: http://www.cnblogs.com/void/archive/2011/08/26/2153928.html 套个spfa。 代码: #include <iostream>#include <cstdio>#i

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

poj 3169 spfa 差分约束

题意: 给n只牛,这些牛有些关系。 ml个关系:fr 与 to 牛间的距离要小于等于 cost。 md个关系:fr 与 to 牛间的距离要大于等于 cost。 隐含关系: d[ i ] <= d[ i + 1 ] 解析: 用以上关系建图,求1-n间最短路即可。 新学了一种建图的方法。。。。。。 代码: #include <iostream>#include

dp算法练习题【8】

不同二叉搜索树 96. 不同的二叉搜索树 给你一个整数 n ,求恰由 n 个节点组成且节点值从 1 到 n 互不相同的 二叉搜索树 有多少种?返回满足题意的二叉搜索树的种数。 示例 1: 输入:n = 3输出:5 示例 2: 输入:n = 1输出:1 class Solution {public int numTrees(int n) {int[] dp = new int

2024网安周今日开幕,亚信安全亮相30城

2024年国家网络安全宣传周今天在广州拉开帷幕。今年网安周继续以“网络安全为人民,网络安全靠人民”为主题。2024年国家网络安全宣传周涵盖了1场开幕式、1场高峰论坛、5个重要活动、15场分论坛/座谈会/闭门会、6个主题日活动和网络安全“六进”活动。亚信安全出席2024年国家网络安全宣传周开幕式和主论坛,并将通过线下宣讲、创意科普、成果展示等多种形式,让广大民众看得懂、记得住安全知识,同时还

2024/9/8 c++ smart

1.通过自己编写的class来实现unique_ptr指针的功能 #include <iostream> using namespace std; template<class T> class unique_ptr { public:         //无参构造函数         unique_ptr();         //有参构造函数         unique_ptr(