NYoj-119-士兵杀敌(3)-RMQ算法

士兵杀敌(三)

时间限制:2000 ms  |  内存限制:65535 KB

难度:5

描写叙述

南将军统率着N个士兵,士兵分别编号为1~N,南将军常常爱拿某一段编号内杀敌数最高的人与杀敌数最低的人进行比較,计算出两个人的杀敌数差值。用这样的方法一方面能鼓励杀敌数高的人,还有一方面也算是批评杀敌数低的人,起到了非常好的效果。

所以,南将军常常问军师小工第i号士兵到第j号士兵中。杀敌数最高的人与杀敌数最低的人之间军功差值是多少。

如今,请你写一个程序,帮小工回答南将军每次的询问吧。

注意。南将军可能询问非常多次。

输入
仅仅有一组測试数据

第一行是两个整数N,Q。当中N表示士兵的总数。

Q表示南将军询问的次数。(1<N<=100000,1<Q<=1000000)

随后的一行有N个整数Vi(0<=Vi<100000000),分别表示每一个人的杀敌数。

再之后的Q行,每行有两个正正数m,n。表示南将军询问的是第m号士兵到第n号士兵。

输出
对于每次询问。输出第m号士兵到第n号士兵之间全部士兵杀敌数的最大值与最小值的差。
例子输入
5 2
1 2 6 9 3
1 2
2 4
例子输出
1
7
#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#include<queue>
#include<stack>
//#define lowbit(i) i&(-i)  //lowbit能够直接define。不须要定义函数
using namespace std;

//那么F(i,0)的值是确定的,就为i这个位置所指的元素值,
//这时我们能够把区间[i,i+2^j-1]平均分为两个区间,由于j>=1的时候该区间的长度始终为偶数。
//能够分为区间[i,i+2^(j-1)-1]和区间[i+2^(j-1)-1,i+2^j-1]。即取两个长度为2^(j-1)的块代替和更新长度为2^j的块。
//那么最小值就是这两个区间的最小值的最小值,动态规划为:F[i,j]=min(F[i,j-1],F[i+2^(j-1),j-1]).
//F[i,j]=max(F[i,j-1],F[i+2^(j-1),j-1]).
//k=ln(j-i+1)/ln2
//同理:最大值就是F[i,j]=max(F[i,j-1],F[i+2^(j-1),j-1]).F[i,j]=min(F[i,j-1],F[i+2^(j-1)+1,j-1]);
//
const int N=100100;
int max_sum[20][N],min_sum[20][N];
void RMQ(int num)
{
	for(int i=1;i!=20;i++)
	{
		for(int j=1;j<=num;j++)
		{
			if(j+(1<<i)-1<=num)  //i<<i==2^i
			{
				max_sum[i][j]=max(max_sum[i-1][j],max_sum[i-1][j+(1<<i>>1)]);
				min_sum[i][j]=min(min_sum[i-1][j],min_sum[i-1][j+(1<<i>>1)]);

			}
		}
	}
}
int main()
{
     int num,query;
     while(~scanf("%d %d",&num,&query))
     {
     	for(int i=1;i<=num;i++)
     	{
     		scanf("%d",&max_sum[0][i]);
     		min_sum[0][i]=max_sum[0][i];
     	}
     	RMQ(num);
     	while(query--)
     	{
     		int begin,end;
     		scanf("%d %d",&begin,&end);
     		int k=(int)(log(end-begin+1.0)/log(2.0));
     		int max1=max(max_sum[k][begin],max_sum[k][end-(1<<k)+1]);
     		int min1=min(min_sum[k][begin],min_sum[k][end-(1<<k)+1]);
     		printf("%d\n",max1-min1);
     	}
     }
	return 0;
}

时间: 2024-10-07 18:20:24

NYoj-119-士兵杀敌(3)-RMQ算法的相关文章

NYOJ 119 士兵杀敌(三) RMQ ST

NYOJ 119 士兵杀敌(三) RMQ ST 题目链接:http://acm.nyist.net/JudgeOnline/problem.php?pid=119 思路: ST在线 预处理O(nlogn) 查询O(1) 运行时间:828ms 代码: #include <iostream> #include <stdio.h> #include <string.h> #include <math.h> using namespace std; const in

nyoj 119士兵杀敌(三)(线段树区间最值查询,RMQ算法)

题目119 题目信息 执行结果 本题排行 讨论区 士兵杀敌(三) 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描写叙述 南将军统率着N个士兵,士兵分别编号为1~N,南将军常常爱拿某一段编号内杀敌数最高的人与杀敌数最低的人进行比較,计算出两个人的杀敌数差值.用这样的方法一方面能鼓励杀敌数高的人,还有一方面也算是批评杀敌数低的人,起到了非常好的效果. 所以,南将军常常问军师小工第i号士兵到第j号士兵中,杀敌数最高的人与杀敌数最低的人之间军功差值是多少. 如今,请你写一个程

NYOJ 119 士兵杀敌(三)(RMQ算法)

采用的的是小牛的写法,蒟蒻第一次写.. RMQ (Range Minimum/Maximum Query)问题是指:对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A中下标在i,j里的最小(大)值,也就是说,RMQ问题是指求区间最值的问题. 这里介绍ST算法:采用动态规划的思想:详见 士兵杀敌(三) 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描述 南将军统率着N个士兵,士兵分别编号为1~N,南将军经常爱拿某一段编号内杀敌数最高的人与杀

nyoj 119 士兵杀敌(三)(RMQ)

士兵杀敌(三) 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描述 南将军统率着N个士兵,士兵分别编号为1~N,南将军经常爱拿某一段编号内杀敌数最高的人与杀敌数最低的人进行比较,计算出两个人的杀敌数差值,用这种方法一方面能鼓舞杀敌数高的人,另一方面也算是批评杀敌数低的人,起到了很好的效果. 所以,南将军经常问军师小工第i号士兵到第j号士兵中,杀敌数最高的人与杀敌数最低的人之间军功差值是多少. 现在,请你写一个程序,帮小工回答南将军每次的询问吧. 注意,南将军可能询问很多

NYOJ 119 士兵杀敌(三)

士兵杀敌(三) 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描述 南将军统率着N个士兵,士兵分别编号为1~N,南将军经常爱拿某一段编号内杀敌数最高的人与杀敌数最低的人进行比较,计算出两个人的杀敌数差值,用这种方法一方面能鼓舞杀敌数高的人,另一方面也算是批评杀敌数低的人,起到了很好的效果. 所以,南将军经常问军师小工第i号士兵到第j号士兵中,杀敌数最高的人与杀敌数最低的人之间军功差值是多少. 现在,请你写一个程序,帮小工回答南将军每次的询问吧. 注意,南将军可能询问很多

nyoj 119 士兵杀敌(三) 【线段树】【单点更新】

题意:... 策略如题. 思路:我们先假设只求某一区间的最大值,我们只需要利用线段树的模板,只需要初始化和询问的时候小小的修改一下,改成祖先结点储存的不再是子节点的和而是两个子节点之间的最大值,这样我们可以求出最大值了,最小值也是这样求. 注意:因为询问的时候既要求最大值又要求最小值,所以要返回结构体. 代码: #include <stdio.h> #include <string.h> #define M 100005 struct node{ int left, right;

NYOJ 116 士兵杀敌 (线段树,区间和)

题目链接:NYOJ 116 士兵杀敌 士兵杀敌(二) 时间限制:1000 ms  |  内存限制:65535 KB 难度:5 描写叙述 南将军手下有N个士兵,分别编号1到N,这些士兵的杀敌数都是已知的. 小工是南将军手下的军师,南将军常常想知道第m号到第n号士兵的总杀敌数,请你帮助小工来回答南将军吧. 南将军的某次询问之后士兵i可能又杀敌q人,之后南将军再询问的时候.须要考虑到新增的杀敌数. 输入 仅仅有一组測试数据 第一行是两个整数N,M,当中N表示士兵的个数(1<N<1000000).M表

NYOJ 116 士兵杀敌(二) (线段树区间求和)

题目链接:NYOJ 116 士兵杀敌(二) 这一个是线段树的入门级水题,本题要求我们给出某个区间的区间和.这个问题和线段树的单点更新还是基本一致的.只要把单点更新中的值覆盖变为值得叠加,这一题便可以轻松解决了.如果不知道线段树的单点更新,请移步:传送门 [代码如下] #include <stdio.h> #include <string.h> #define MAXN 1<<21 //lg100000 约等于 21 typedef struct{ int left,ri

NYOJ 116 士兵杀敌(二)【线段树 单点更新】

题意:题意很清楚: 策略:如题. 这道题就是简单的线段树应用,据说还可以用树状数组来做,等我学了之后在说吧. 代码: #include<stdio.h> #include<string.h> #define LC l, m, rt<<1 #define RC m+1, r, rt<<1|1 #define LL long long #define MAXN 1000000 LL sum[MAXN<<2]; void PushUp(int rt)

NYIST 119 士兵杀敌(三)

士兵杀敌(三)时间限制:2000 ms | 内存限制:65535 KB难度:5 描述南将军统率着N个士兵,士兵分别编号为1~N,南将军经常爱拿某一段编号内杀敌数最高的人与杀敌数最低的人进行比较,计算出两个人的杀敌数差值,用这种方法一方面能鼓舞杀敌数高的人,另一方面也算是批评杀敌数低的人,起到了很好的效果. 所以,南将军经常问军师小工第i号士兵到第j号士兵中,杀敌数最高的人与杀敌数最低的人之间军功差值是多少. 现在,请你写一个程序,帮小工回答南将军每次的询问吧. 注意,南将军可能询问很多次. 输入