• 倍增(小试牛刀)


    二分每次折半,倍增每次2的倍数

    原理先存储每个小区间的最值也就是初始化,之后直接查询

    1. 把数列按倍增分成小区间

    对数列的每个元素,把从它开始的数列分成长度为1、2、4、8、…的小区间。下图给出了一个分区的例子,它按小区间的长度分成了很多组。

    • 第 1 组是长度为 1 的小区间,有 n 个小区间,每个小区间有 1 个元素;
    • 第 2 组是长度为 2 的小区间,有 n 个小区间,每个小区间有 2 个元素;
    • 第 3 组是长度为 4 的小区间,有 n 个小区间,每个小区间有 4 个元素;
    • ...共有 logn (2为底)组。

    每组的小区间的最值,可以从前一组递推而来。例如第 3 组 {4, 7, 9, 6} 的最值,可以从第 2 组 {4, 4,7}、{9,6} 的最值递推得到。

    所以我们定义 dp[s][k],表示左端点是 s,区间长度为 2^k 的区间最值。它的递推关系是:(1<<(k-1) 等于 2^{k-1})

    所以:dp[s][k]=min(dp[s][k−1],dp[s+1<<(k−1)][k−1])

    dp[s][k]当前组,是由后面两组推导出来的

    图中的每一组都需计算 n 次,共 logn组,总计算量是 O(nlogn)。

    2.查询

    查询区间[L,R]的最值,L是左端点,R是右端点,区间长度len=R-L+1,区间长度为 2^k 的最值2^k=len

    所以k=log(R-L+1)  (2为底,注意Java中的Math.log(n)是以e为底

    如果以10为底的话 k=log(R-L+1)/log(2.0)           (换底公式)

    最后给出区间 [L, R]最小值的计算公式,等于覆盖它的两个小区间的最小值:

    min(dp[L][k],dp[R−(1<

    3.题目练习

    https://www.lanqiao.cn/problems/1205/learning/

    区间最大值

    给定一个长度为 N 的数组 a,其值分别为 a1,a2,...,aN 
    现有 Q 个询问,每个询问包含一个区间,请回答该区间的最大值为多少。

    输入描述
    输入第 1行包含两个正整数 N,Q,分别表示数组 a的长度和询问的个数。

    第 2 行包含 N 个非负整数 a1,a2,...,aN ,表示数组 a 元素的值。

    后面每行表示一个询问,每个询问包含两个整数 L,R,表示区间的左右端点

    1<=N,Q<=5*10^5   -10^9<=ai<=10^9
    输出描述
    输出共 Q 行,每行包含一个整数,表示相应询问的答案。

    输入输出样例
    示例 1
    输入

    5 5
    1 2 3 4 5
    1 1 
    1 2 
    1 3
    3 4
    2 5

    输出

    1
    2
    3
    4
    5

    1. import java.util.Scanner;
    2. public class Main {
    3. static int N=500000;
    4. static int n,m;
    5. static int L,R;
    6. static int a[]=new int[N];
    7. //第二个参数是k 设为40即可,也可其他
    8. static int dp_max[][]=new int[N][40];
    9. public static void main(String[] args) {
    10. Scanner sc = new Scanner(System.in);
    11. n=sc.nextInt();
    12. m=sc.nextInt();
    13. for(int i=1;i<=n;i++) {
    14. a[i]=sc.nextInt();
    15. }
    16. st_init();
    17. for(int j=1;j<=m;j++){
    18. L=sc.nextInt();
    19. R=sc.nextInt();
    20. System.out.println(st_query(L,R));
    21. }
    22. }
    23. // 把数列按倍增分成小区间
    24. public static void st_init(){
    25. for(int i=1;i<=n;i++) //初始化区间长度为1时的值
    26. dp_max[i][0]=a[i];
    27. int p=(int)(Math.log10(n)/Math.log10(2.0));
    28. for(int k=1;k<=p;k++) //倍增计算小区间。先算小区间,再算大区间,逐步递推
    29. for(int s=1;s+(1<1;s++)
    30. dp_max[s][k]=Math.max(dp_max[s][k-1], dp_max[s+(1<<(k-1))][k-1]);
    31. }
    32. //查询
    33. public static int st_query(int L,int R){
    34. int k =(int)(Math.log10(R-L+1)/Math.log10(2.0));
    35. return Math.max(dp_max[L][k],dp_max[R-(1<1][k]);
    36. }
    37. }

    https://www.lanqiao.cn/problems/1375/learning/

    同理

    1. import java.util.Scanner;
    2. public class Main {
    3. static int N=500000;
    4. static int n,m;
    5. static int L,R;
    6. static int a[]=new int[N];
    7. //第二个参数是k 设为40即可,也可其他
    8. static int dp_min[][]=new int[N][40];
    9. public static void main(String[] args) {
    10. Scanner sc = new Scanner(System.in);
    11. n=sc.nextInt();
    12. m=sc.nextInt();
    13. for(int i=1;i<=n;i++) {
    14. a[i]=sc.nextInt();
    15. }
    16. st_init();
    17. for(int i=1;i<=n-m+1;i++){
    18. System.out.println(st_query(i,i+m-1));
    19. }
    20. }
    21. // 把数列按倍增分成小区间
    22. public static void st_init(){
    23. for(int i=1;i<=n;i++) //初始化区间长度为1时的值
    24. dp_min[i][0]=a[i];
    25. int p=(int)(Math.log10(n)/Math.log10(2.0));
    26. for(int k=1;k<=p;k++) //倍增计算小区间。先算小区间,再算大区间,逐步递推
    27. for(int s=1;s+(1<1;s++)
    28. dp_min[s][k]=Math.min(dp_min[s][k-1], dp_min[s+(1<<(k-1))][k-1]);
    29. }
    30. //查询
    31. public static int st_query(int L,int R){
    32. int k =(int)(Math.log10(R-L+1)/Math.log10(2.0));
    33. return Math.min(dp_min[L][k],dp_min[R-(1<1][k]);
    34. }
    35. }

  • 相关阅读:
    【函数式编程】Lambda、Stream、Optional、方法引用、并行流
    GPT-4V的图片识别和分析能力
    Python版中秋佳节月饼抢购脚本
    ruby ftp封装实例详解
    逻辑运算介绍
    次时代终端工具:WindTerm(含下载)
    [RK3568 Android11]Wifi 连接过程
    基于JAVA的会议管理系统参考【数据库设计、源码、开题报告】
    学习登录接口过程中的踩坑记录
    Java集合常见面试题
  • 原文地址:https://blog.csdn.net/qq_58631644/article/details/128050010