倍增(小试牛刀)

Posted 猪八戒1.0

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了倍增(小试牛刀)相关的知识,希望对你有一定的参考价值。

二分每次折半,倍增每次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<<k)+1][k]);             (1<<k也就是2^k)

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


import java.util.Scanner;

public class Main 
	static int N=500000;
	static int n,m;
	static int L,R;
	static int a[]=new int[N];
	//第二个参数是k 设为40即可,也可其他
	static int dp_max[][]=new int[N][40];
	
	public static void main(String[] args) 
		Scanner sc = new Scanner(System.in);
		n=sc.nextInt();
		m=sc.nextInt();
		for(int i=1;i<=n;i++) 
			a[i]=sc.nextInt();
		
		st_init();
		for(int j=1;j<=m;j++)		
	        L=sc.nextInt();
	        R=sc.nextInt();	  
	        System.out.println(st_query(L,R));
	    
		
	


	// 把数列按倍增分成小区间
	public static void st_init()
	    for(int i=1;i<=n;i++)   //初始化区间长度为1时的值
	        dp_max[i][0]=a[i];  
	    	int p=(int)(Math.log10(n)/Math.log10(2.0));     
	    
	    for(int k=1;k<=p;k++) //倍增计算小区间。先算小区间,再算大区间,逐步递推
	        for(int s=1;s+(1<<k)<=n+1;s++)
	            dp_max[s][k]=Math.max(dp_max[s][k-1], dp_max[s+(1<<(k-1))][k-1]);
	
	//查询
	public static int st_query(int L,int R)
		int k =(int)(Math.log10(R-L+1)/Math.log10(2.0));           
	    return Math.max(dp_max[L][k],dp_max[R-(1<<k)+1][k]);
	

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

同理



import java.util.Scanner;

public class Main 
	static int N=500000;
	static int n,m;
	static int L,R;
	static int a[]=new int[N];
	//第二个参数是k 设为40即可,也可其他
	static int dp_min[][]=new int[N][40];
	
	public static void main(String[] args) 
		Scanner sc = new Scanner(System.in);
		n=sc.nextInt();
		m=sc.nextInt();
		for(int i=1;i<=n;i++) 
			a[i]=sc.nextInt();
		
		st_init();
		for(int i=1;i<=n-m+1;i++)		
	        System.out.println(st_query(i,i+m-1));
	    
		
	


	// 把数列按倍增分成小区间
	public static void st_init()
	    for(int i=1;i<=n;i++)   //初始化区间长度为1时的值
	        dp_min[i][0]=a[i];  
	    	int p=(int)(Math.log10(n)/Math.log10(2.0));     
	    
	    for(int k=1;k<=p;k++) //倍增计算小区间。先算小区间,再算大区间,逐步递推
	        for(int s=1;s+(1<<k)<=n+1;s++)
	            dp_min[s][k]=Math.min(dp_min[s][k-1], dp_min[s+(1<<(k-1))][k-1]);
	
	//查询
	public static int st_query(int L,int R)
		int k =(int)(Math.log10(R-L+1)/Math.log10(2.0));           
	    return Math.min(dp_min[L][k],dp_min[R-(1<<k)+1][k]);
	

以上是关于倍增(小试牛刀)的主要内容,如果未能解决你的问题,请参考以下文章

WebSocket小试牛刀

博客园设置访问数量统计-小试牛刀

模板st表

Codeforces 827D Best Edge Weight 倍增 + 并查集 || 倍增 + 压倍增标记 (看题解)

LCA之倍增简单讲解

倍增算法