news 2026/4/23 13:18:20

信奥赛C++提高组csp-s之倍增算法

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
信奥赛C++提高组csp-s之倍增算法

信奥赛C++提高组csp-s之倍增算法

倍增算法核心思想讲解

1. 什么是倍增?

“倍增”,顾名思义,就是成倍地增加。它的核心思想是:不是一步一步地处理问题,而是将每一步的“步长”以2的幂次(1, 2, 4, 8…)进行跳跃式处理

2. 为什么倍增有效?
  • 高效性:通过二进制划分,可以将一个线性过程的时间复杂度从 O(N) 优化到 O(logN)。
  • 可行性:任何整数都可以用二进制数表示。这意味着,从起点到终点的任意一个长度,我们都可以通过2^k1 + 2^k2 + ...的跳跃方式到达。
  • 预处理:倍增算法通常需要先进行预处理,计算出一些“跳跃点”的信息,以便在查询时能够快速组合。
3. 倍增的通用步骤
  1. 定义状态f[i][j]通常表示从i这个点出发,走2^j步(或者进行2^j次操作)后所到达的状态。这个“状态”可以是到达的位置、区间的最值、区间和等。
  2. 预处理(DP填充):这是算法的关键。我们利用递推关系来填充这个f数组(也称为ST表、DP表)。
    • 边界条件f[i][0]表示从i走 1 (2^0) 步后的状态。这是初始的、已知的数据。
    • 递推公式f[i][j] = f[ f[i][j-1] ][j-1]。这个公式是倍增的灵魂!它的意思是:i2^j步到达的点,等价于从i先走2^(j-1)步,到达一个中间点f[i][j-1],然后再从这个中间点走2^(j-1)
  3. 查询/执行:对于一次查询,比如“从点uk步会到哪?”,我们将k分解成二进制。例如k = 13 = 8 + 4 + 1(二进制1101)。那么我们就依次走 8 步、4 步、1 步。每一步的跳跃都可以直接从我们预处理好的f数组中 O(1) 获取。

案例:ST 表 & RMQ 问题

题目描述

给定一个长度为N NN的数列,和 $ M $ 次询问,求出每一次询问的区间内数字的最大值。

输入格式

第一行包含两个整数N , M N,MN,M,分别表示数列的长度和询问的个数。

第二行包含N NN个整数(记为a i a_iai),依次表示数列的第i ii项。

接下来M MM行,每行包含两个整数l i , r i l_i,r_ili,ri,表示查询的区间为[ l i , r i ] [l_i,r_i][li,ri]

输出格式

输出包含M MM行,每行一个整数,依次表示每一次询问的结果。

输入输出样例 #1
输入 #1
8 8 9 3 1 7 5 6 0 8 1 6 1 5 2 7 2 6 1 8 4 8 3 7 1 8
输出 #1
9 9 7 7 9 8 7 9
说明/提示

对于30 % 30\%30%的数据,满足1 ≤ N , M ≤ 10 1\le N,M\le 101N,M10

对于70 % 70\%70%的数据,满足1 ≤ N , M ≤ 10 5 1\le N,M\le {10}^51N,M105

对于100 % 100\%100%的数据,满足1 ≤ N ≤ 10 5 1\le N\le {10}^51N1051 ≤ M ≤ 2 × 10 6 1\le M\le 2\times{10}^61M2×106a i ∈ [ 0 , 10 9 ] a_i\in[0,{10}^9]ai[0,109]1 ≤ l i ≤ r i ≤ N 1\le l_i\le r_i\le N1liriN

AC代码

#include<bits/stdc++.h>usingnamespacestd;constintN=1e5+10;// 定义最大数组长度intn,m;// n: 数列长度, m: 询问个数inta[N];// 存储原始数列intst[N][17];// ST表,st[i][j]表示从位置i开始,长度为2^j的区间最大值intmain(){cin>>n>>m;// 读入数列并初始化ST表第一层for(inti=1;i<=n;i++){scanf("%d",&a[i]);st[i][0]=a[i];// 长度为1的区间最大值就是元素本身}// 构建ST表for(intj=1;j<=17;j++){// j表示区间长度为2^jfor(inti=1;i+(1<<j)-1<=n;i++){// i为区间起点// 将区间[i, i+2^j-1]分成两个长度为2^(j-1)的子区间// 取两个子区间最大值的较大者作为当前区间的最大值st[i][j]=max(st[i][j-1],st[i+(1<<(j-1))][j-1]);}}// 处理每个查询while(m--){intl,r;scanf("%d%d",&l,&r);// 计算区间长度的对数(向下取整)ints=log2(r-l+1);// 查询区间最大值:将区间分成可能有重叠的两部分// [l, l+2^s-1] 和 [r-2^s+1, r]intans=max(st[l][s],st[r-(1<<s)+1][s]);printf("%d\n",ans);}return0;}

功能分析

使用ST表(Sparse Table)解决区间最大值查询(RMQ)

核心思想:
  • 预处理:构建一个二维数组st,其中st[i][j]存储从位置i开始,长度为2^j的区间内的最大值
  • 查询:对于任意区间[l, r],可以将其分解为两个可能有重叠的区间,取这两个区间最大值的较大者
算法步骤:
  1. 初始化

    • st[i][0] = a[i](长度为1的区间最大值就是元素本身)
  2. 构建ST表

    • 使用动态规划思想,st[i][j] = max(st[i][j-1], st[i+2^(j-1)][j-1])
    • 即将大区间分成两个小区间,取两者的最大值
  3. 查询处理

    • 对于区间[l, r],计算s = log2(r-l+1)(区间长度的对数)
    • 查询结果 =max(st[l][s], st[r-2^s+1][s])
复杂度分析:
  • 预处理:O(N log N)
  • 单次查询:O(1)
  • 总复杂度:O(N log N + M)
优势:
  • 查询速度极快(O(1)),适合处理大量查询
  • 代码简洁,实现相对容易
适用场景:
  • 静态数据(数据不修改)
  • 查询次数远大于数据修改次数
  • 需要快速响应大量区间查询

更多系列知识,请查看专栏:《信奥赛C++提高组csp-s知识详解及案例实践》:
https://blog.csdn.net/weixin_66461496/category_13113932.html


各种学习资料,助力大家一站式学习和提升!!!

#include<bits/stdc++.h>usingnamespacestd;intmain(){cout<<"########## 一站式掌握信奥赛知识! ##########";cout<<"############# 冲刺信奥赛拿奖! #############";cout<<"###### 课程购买后永久学习,不受限制! ######";return0;}

1、csp信奥赛高频考点知识详解及案例实践:

CSP信奥赛C++动态规划:
https://blog.csdn.net/weixin_66461496/category_13096895.html点击跳转

CSP信奥赛C++标准模板库STL:
https://blog.csdn.net/weixin_66461496/category_13108077.html 点击跳转

信奥赛C++提高组csp-s知识详解及案例实践:
https://blog.csdn.net/weixin_66461496/category_13113932.html

2、csp信奥赛冲刺一等奖有效刷题题解:

CSP信奥赛C++初赛及复赛高频考点真题解析(持续更新):https://blog.csdn.net/weixin_66461496/category_12808781.html 点击跳转

CSP信奥赛C++一等奖通关刷题题单及题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12673810.html 点击跳转

3、GESP C++考级真题题解:

GESP(C++ 一级+二级+三级)真题题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12858102.html 点击跳转

GESP(C++ 四级+五级+六级)真题题解(持续更新):https://blog.csdn.net/weixin_66461496/category_12869848.html 点击跳转


GESP(C++ 七级+八级)真题题解(持续更新):
https://blog.csdn.net/weixin_66461496/category_13117178.html

4、CSP信奥赛C++竞赛拿奖视频课:

https://edu.csdn.net/course/detail/40437 点击跳转

· 文末祝福 ·

#include<bits/stdc++.h>usingnamespacestd;intmain(){cout<<"跟着王老师一起学习信奥赛C++";cout<<" 成就更好的自己! ";cout<<" csp信奥赛一等奖属于你! ";return0;}
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 9:58:04

Wan2.2实战教程:基于ComfyUI的工作流配置与调试详细步骤

Wan2.2实战教程&#xff1a;基于ComfyUI的工作流配置与调试详细步骤 1. 教程目标与适用场景 随着AIGC技术的快速发展&#xff0c;文本到视频&#xff08;Text-to-Video&#xff09;生成已成为内容创作领域的重要工具。Wan2.2-I2V-A14B作为通义万相推出的高效视频生成模型&…

作者头像 李华
网站建设 2026/4/22 19:37:43

STM32调试利器:STLink驱动安装深度剖析

STM32调试从“连不上”到“秒识别”&#xff1a;STLink驱动安装全链路实战指南 你有没有过这样的经历&#xff1f; 新焊好一块STM32板子&#xff0c;兴冲冲插上STLink&#xff0c;打开IDE准备烧录程序——结果设备管理器里赫然显示一个黄色感叹号&#xff1a;“ STM Device …

作者头像 李华
网站建设 2026/4/23 11:34:21

Live Avatar部署教程:4卡24GB配置下的参数调优技巧

Live Avatar部署教程&#xff1a;4卡24GB配置下的参数调优技巧 1. 引言 Live Avatar是由阿里巴巴联合多所高校共同开源的一款先进数字人生成模型&#xff0c;旨在通过文本、图像和音频输入驱动高保真虚拟人物视频的生成。该模型基于14B参数规模的DiT&#xff08;Diffusion in…

作者头像 李华
网站建设 2026/4/23 11:35:55

AI Agent 在汽车上的典型应用场景,研发入门

汽车领域&#xff0c;AI Agent 通常以 “多智能体协同” 的形式存在。从近两年开始&#xff0c;AI Agent 在汽车上正从单点功能升级为全链路场景化智能中枢。 系统总结了AI Agent 在汽车行业的应用&#xff0c;覆盖智能座舱、自动驾驶、车联网服务与车辆运维四大领域&#xff0…

作者头像 李华
网站建设 2026/4/10 19:25:24

解决Module Federation中的NG_VALUE_ACCESSOR问题

引言 在现代Web开发中,模块联邦(Module Federation)技术被广泛应用于微前端架构中。特别是在使用NX、Angular和Ionic构建应用时,模块联邦可以帮助我们实现代码共享和独立部署。然而,某些配置问题可能会导致意想不到的错误,比如ControlValueAccessor缺失的问题。今天我们…

作者头像 李华
网站建设 2026/4/23 11:34:48

Telerik Reporting 2023 升级指南:解决前后端兼容性问题

随着 Telerik Reporting 的不断更新,开发人员在升级时常常会遇到一些兼容性问题。本文将详细讨论在升级到 Telerik Reporting 2023 版本时,前后端如何协调工作,解决常见的问题,并提供实际案例。 前言 Telerik Reporting 是一个强大的报表解决方案,广泛应用于企业级应用中…

作者头像 李华