An easy problem A

Time Limit: 1000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others)

Problem Description

N个数排成一列,Q个询问,每次询问一段区间内的数的极差是多少。

Input

第一行两个整数N(1≤N≤50000),Q(1≤Q≤200000)。接下来一行N个整数a1 a2 a3 ….an,(1≤ai≤1000000000)。接下来Q行,每行两个整数L,R(1≤L≤R≤N)。

Output

对于每个询问输出一行,一个整数表示区间内的极差。

Sample Input

5 3

3 2 7 9 10

1 5

2 3

3 5

Sample Output

8

5

3


解题心得:

一、线段树

1. 这是一个最简单的线段树的模板,主要是入门理解线段树。当数据量比较大,需要多次询问,多次修改维护数据时,就需要使用到线段树的数据结构。线段树的每个节点代表的是一段线段,每次修改这个线段,就可以由此对每个区间进行维护,缩减时间复杂度和空间复杂度。

2. 一般线段树的空间复杂度是O(4n),建立线段树时间复杂度是O(logn),向上更新的时间复杂度是O(logn),询问的时间复杂度是O(logn),如果不使用线段树就遍历硬怼时间复杂度是O(n^2)。线段树是根节点将根节点代表的线段均分为两个线段,分别是左子节点和右子节点,然后一直分下去直到一个线段的长度为1(只有一个数)对该点进行初始赋值,然后向上维护得到每一个线段的需要得到值,线段树就是尽量避免对单个的点进行操作,而是对区间进行维护,这个就避免一个点一个点的去比较,可以直接从区间上面进行操作,树形结构可以很方便的进行维护操作。

3. 就本题来说就是将整个数列按照完美二叉树的样子直接分下去,一直分到区间长度为1,这时这个节点的最大值和最小值都是他自己,然后再向上更新,向上更新的时候父节点的最大值和最小值又他的两个子节点得到,一直向上更新,时间复杂度是nlogn。

二、

  1. 这个还可以使用RMQ算法,即区间最值查询算法,这个算法就是在区间中使用动态规划对这个区间进行预处理,倍增思想。
  2. 动态规划时创建一个dp[i][j],代表以I为起点长度为2^j次方的区间最值,它的转移方程式是dp[I][j] = max(dp[i][j-1],dp[i+2^(j-1)][j-1]),它的意思是从i开始长度为2^j区间的最大值和i+2^(j-1)和长度为2^(j-1)区间的最大值(倍增算法的基础运用)。
  3. 当要求在I到j的最大值的时候可以用i到i + 2 ^ x(i+ 2 ^ x < j ) 和 j - 2 ^ x到j中的最大值。(MAX = max(max(num[i],num[I+2^x]),max(num[j-2^x],num[j]))).
  4. 其实仔细想想,线段数和RMQ的算法是相似的,只是实现的方法不一样,但是思想是一样的(一个二分,一个倍增)。都是将点联系起来用区间来表示。

线段树代码
#include<bits/stdc++.h>
using namespace std;
const int maxn = 5e4+100;
struct node
{
int l,r,Max,Min;
}tree[4*maxn];//经计算生成线段树消耗的空间小于4n
int n,m,num[maxn]; //向上更新
void pushup(int root)
{
tree[root].Max = max(tree[root*2].Max,tree[root*2+1].Max);
tree[root].Min = min(tree[root*2].Min,tree[root*2+1].Min);
} //生成线段树
void build_tree(int l,int r,int root)
{
int mid;
tree[root].l = l;
tree[root].r = r;
if(l == r)
{
tree[root].Min = tree[root].Max = num[l];
return ;
}
mid = (l+r)/2;
build_tree(l,mid,root*2);//左孩子
build_tree(mid+1,r,root*2+1);//右孩子
pushup(root);//在线段树已经建成了之后开始向上更新
} //得到最大值
int queryMax(int l,int r,int root)
{
int mid;
if(tree[root].l == l && tree[root].r == r)
return tree[root].Max;
mid = (tree[root].l + tree[root].r) / 2;
if(r <= mid)
return queryMax(l,r,root*2);
else if(l > mid)
return queryMax(l,r,root*2+1);
else return max(queryMax(l,mid,root*2),queryMax(mid+1,r,root*2+1));
} //得到最小值
int queryMin(int l,int r,int root)
{
int mid;
if(tree[root].l == l && tree[root].r == r)
return tree[root].Min;
mid = (tree[root].l + tree[root].r) / 2;
if(r <= mid)
return queryMin(l,r,root*2);
else if(l > mid)
return queryMin(l,r,root*2+1);
else return min(queryMin(l,mid,root*2),queryMin(mid+1,r,root*2+1));
} int main()
{
while(scanf("%d%d",&n,&m) != EOF)
{
for(int i=1;i<=n;i++)
scanf("%d",&num[i]);
build_tree(1,n,1);
while(m--)
{
int a,b;
scanf("%d%d",&a,&b);
`
printf("%d\n",queryMax(a,b,1)-queryMin(a,b,1));
}
}
}

RMQ算法

#include<bits/stdc++.h>
using namespace std;
const int maxn = 5e4+100;
int num[maxn];
int n,m; struct Dp
{
int Min;
int Max;
} dp[maxn][20]; void DP()
{
for(int j=0; j<20; j++)
{
for(int i=1; i<=n; i++)
{
if(j == 0)
{
dp[i][j].Max = dp[i][j].Min = num[i];//当就只有一个数的时候最大值和最小值都是这个数
continue;
}
if(i + (1<<(j - 1)) > n)//不能超出限制
continue;
int k = 1<<(j-1);
dp[i][j].Max = max(dp[i][j-1].Max,dp[i+k][j-1].Max);//状态转移方程式
dp[i][j].Min = min(dp[i][j-1].Min,dp[i+k][j-1].Min);
}
}
}
int main()
{
while(scanf("%d%d",&n,&m) != EOF)
{
memset(dp,0,sizeof(dp)); for(int i=1; i<=n; i++)
scanf("%d",&num[i]);
DP();
while(m--)
{
int l,r,l1,r1;
scanf("%d%d",&l,&r);
//这是将要求的区间化成两重合的可以用dp[i][j]的区间来表示,求最大值
r1 = log2(r-l+1);
l1 = r - (1<<r1) + 1;
printf("%d\n",max(dp[l][r1].Max,dp[l1][r1].Max)-min(dp[l][r1].Min,dp[l1][r1].Min));
}
}
return 0;
}

线段树:CDOJ1591-An easy problem A (RMQ算法和最简单的线段树模板)的更多相关文章

  1. 线段树:CDOJ1592-An easy problem B (线段树的区间合并)

    An easy problem B Time Limit: 2000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Pr ...

  2. 线段树:CDOJ1597-An easy problem C(区间更新的线段树)

    An easy problem C Time Limit: 4000/2000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Pr ...

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

    题目119 题目信息 执行结果 本题排行 讨论区 士兵杀敌(三) 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描写叙述 南将军统率着N个士兵,士兵分别编号为1~N,南将军常 ...

  4. hdu 5475 An easy problem(暴力 || 线段树区间单点更新)

    http://acm.hdu.edu.cn/showproblem.php?pid=5475 An easy problem Time Limit: 8000/5000 MS (Java/Others ...

  5. HDU 5475 An easy problem 线段树

    An easy problem Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pi ...

  6. UESTC 1591 An easy problem A【线段树点更新裸题】

    An easy problem A Time Limit: 2000/1000MS (Java/Others)     Memory Limit: 65535/65535KB (Java/Others ...

  7. HDU 5475:An easy problem 这题也能用线段树做???

    An easy problem Time Limit: 8000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

  8. hud-5475 An easy problem(线段树)

    http://acm.hdu.edu.cn/showproblem.php?pid=5475 An easy problem Time Limit: 8000/5000 MS (Java/Others ...

  9. HDU 5475An easy problem 离线set/线段树

    An easy problem Time Limit: 8000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

随机推荐

  1. 【转载】Ubuntu16.04安装最新版nodejs

    安装最新版nodejs 更新ubuntu软件源 sudo apt-get update sudo apt-get install -y python-software-properties softw ...

  2. ruby Iconv.iconv编码方法

    #定义一个UTF-8=>GBK的方法def encoding inStr    Iconv.iconv("GBK","UTF-8",inStr)end#定 ...

  3. 洛谷-P3927 SAC E#1 - 一道中档题 Factorial

    原址 题目背景 数据已修改 SOL君(炉石主播)和SOL菌(完美信息教室讲师)是好朋友. 题目描述 SOL君很喜欢阶乘.而SOL菌很喜欢研究进制. 这一天,SOL君跟SOL菌炫技,随口算出了n的阶乘. ...

  4. Linq 根据list属性去重复

    s.Where((x, i) => s.FindIndex(z => z.ArticleTitle == x.ArticleTitle) == i).ToList();

  5. QrenCode : linux命令行下生成二维码图片

    原文链接:http://wowubuntu.com/qrencode.html # 作者:riku/ / 本文采用CC BY-NC-SA 2.5协议授权,转载请注明本文链接. 对于二维码大家应该并不陌 ...

  6. ORACLE中能否找到未提交事务的SQL语句

      在Oracle数据库中,我们能否找到未提交事务(uncommit transactin)的SQL语句或其他相关信息呢?  关于这个问题,我们先来看看实验测试吧.实践出真知. 首先,我们在会话1(S ...

  7. python+selenium之多表单切换

    在Web应用中经常会遇到fram/iframe表单嵌套页面的应用,WebDriver只能在一个页面上对元素识别与定位,对于fram/iframe表单内嵌套页面上的元素无法直接定位.这是需要通过swit ...

  8. Vue.js-this详解

    this this 指向并不是在函数定义的时候确定的,而是在调用的时候确定的.换句话说,函数的调用方式(直接调用.方法调用.new调用.bind.call.apply.箭头函数)决定了 this 指向 ...

  9. Nginx+Keepalived负载均衡+后端LNMP网站集群

    Centos6.4 x86,4台,地址是10.10.10.11-14,vip给的100,目标是在13和14安装nginx+keepalived,11和12安装nginx+mysql+php,做为web ...

  10. 洛谷 P2323 [HNOI2006]公路修建问题

    题目描述 输入输出格式 输入格式: 在实际评测时,将只会有m-1行公路 输出格式: 输入输出样例 输入样例#1: 4 2 5 1 2 6 5 1 3 3 1 2 3 9 4 2 4 6 1 3 4 4 ...