范围最小值问题(Range Minium Query,RMQ)---RMQ问题

一、一维问题

给出一个n个元素的数组A1,A2,...,An,

设计一个数据结构,

支持查询操作Query(L,R):计算min(AL,AL+1,...AR

显然,

用一个循环来计算最小值

显然不够快,

即使是前缀和的思想也不能提高效率!

那么,

实践中最常用的是Tarjan的Sparse-Table算法(就是ST表)

预处理时间:O(nlogn)

查询时间:O(1)

(这个算法非常好写,而且还不容易出错)

(其实:RMQ问题可以做到O(n)预处理,O(1)查询,但,我不会,嘻嘻,蒟蒻本质)

--------------------------------------------------------

ST表:

用的是倍增的思想

用来求区间最值问题

(ST表即可以求最大值,又可以求最小值;但RMQ问题只需要求区间最小值,所以,可以说是,用ST这种算法来解决RMQ问题)

令d(i,j)表示从i开始的,长度为2j的一段元素中的最小值;

那么,就可以用递推的方法计算

  d(i,j)=min{d(i,j-1),d(i+2j-1,j-1)}

原理如下

时间复杂度分析:

  2j<=n;因此d数组的元素个数不超过nlogn,而每一项都可以在常数时间计算完毕,故总时间为O(nlogn)

代码:

void RMQ_init(const vector<int>& A)
{
int n = A.size();
for(int i = ;i < n;i++)
d[i][] = A[i];
for(int j = ;( << j) <= n;j++)
for(int i = ;i + (<<j) - < n;i++)
d[i][j] = min(d[i][j-],d[i + (<<(j-))][j-]);
}

上面这是预处理的操作

下面就是查询操作

由于并不是每个数都恰好是2的k次方

所以只能退而求其次

令k满足2k<=R-L+1的最大整数

那么

以L开头,以R结尾的两个长度为2k的区间合起来机覆盖了查询区间[L,R]。

由于是取最小值

所以有些元素

重复考虑了几遍也没关系

(但如果是累加,重复元素就是不允许的了)

原理如下

查询的代码:

int RMQ(int L,int R)
{
int k = ;
while(<<(k+) <= R-L+)
k++;//如果2^(k+1)<=R-L+1.那么k还可以加1
//这步保证k是取了它的最大值,即下面的两个范围是把全部部分都覆盖了
return min(d[L][k],d[r-(<<k)+][k]);
}

二、二维问题

二维RMQ问题就是求一个矩阵N*M中的一个小块矩阵内的最值问题.其中dmin[i][j][ii][jj]=x表示以(i, j)为左上角,以(i+(1<<ii)-1, j+(1<<jj)-1 )为右下角的矩阵内的最小值.dmax的值类似.

下面dmin[i][j][ii][jj]的值如何求呢?首先我们知道dmin[i][j][0][0]的值就是v[i][j],而假设dmin[i][j][ii][jj]中的ii不为0,那么dmin[i][j][ii][jj]= min(dmin[i][j][ii-1][jj], dmin[i+(1<<ii)][j][ii-1][jj] );如果ii为0,那么就按jj来求.

其实上面的求法就是等于把二维问题转变为一维问题来求解.

下面我们讨论如何查询结果.

对于一个以(x1, y1)为左上角,以(x2, y2)为右下角的矩形,如何求它的最小值和最大值呢?下面假设我们求最小值:

我们把(x1,y1)与(x2,y2)构成的矩形分成四小块,这四小块可能有重合部分,但是它们共同构成了目标矩形:

dmin[x1][y1][ii][jj]

dmin[x1][y2-(1<<jj)+1][ii][jj]

dmin[x2-(1<<ii)+1][y1][ii][jj]

dmin[x2-(1<<ii)+1][y2-(1<<jj)+1][ii][jj]

(自己想象下上面4小块是怎么样的?)

temp 1=min(dmin[x1][y1][ii][jj] , dmin[x1][y2-(1<<jj)+1][ii][jj])

temp 2=min(dmin[x2-(1<<ii)+1][y1][ii][jj] ,dmin[x2-(1<<ii)+1][y2-(1<<jj)+1][ii][jj] )

最终结果是min(temp1, temp2);

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int MAXN=+; int dmax[MAXN][];
int dmin[MAXN][]; void initmax(int n,int d[])//初始化最大值查询
{
for(int i=; i<=n; i++)
dmax[i][]=d[i];
for(int j= ; (<<j)<=n ; j++)
for(int i=; i+(<<j)- <=n; i++)
dmax[i][j]=max(dmax[i][j-],dmax[i+(<<(j-))][j-]);
}
int getmax(int L,int R)//查询最大值
{
int k=;
while((<<(k+))<=R-L+)k++;
return max(dmax[L][k] , dmax[R-(<<k)+][k]);
} void initmin(int n,int d[])//初始化最小值查询
{
{
for(int i=; i<=n; i++)
dmin[i][]=d[i];
for(int j=; (<<j)<=n; j++)
for(int i=;i+(<<j)-<=n;i++)
dmin[i][j]= min( dmin[i][j-],dmin[i+(<<(j-))][j-] );
}
int getmin(int L,int R)//查询最小值
{
int k=;
while( (<<(k+)) <=R-L+)k++;
return min(dmin[L][k],dmin[R-(<<k)+][k]);
}

RMQ问题--范围最小值问题的更多相关文章

  1. hdu 3183 A Magic Lamp RMQ ST 坐标最小值

    hdu 3183 A Magic Lamp RMQ ST 坐标最小值 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3183 题目大意: 从给定的串中挑 ...

  2. RMQ 区间最大值 最小值查询

    /*RMQ 更新最小值操作 By:draymonder*/ #include <iostream> #include <cstdio> using namespace std; ...

  3. RMQ 区间最大值最小值 最频繁次数

    区间的最大值和最小值 #include <cstdio> #include <cstring> #include <cmath> #include <iost ...

  4. UVa 11235 RMQ

    首先讲一下RMQ算法的意思. RMQ(Range Minimum Query,RMQ)范围最小值,给出一个n个元素的数组,计算min(A[L],A[L+1],...,A[R-1],A[R]): 这里运 ...

  5. hdu3183 RMQ

    A Magic Lamp Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tot ...

  6. POJ 3264 RMQ水题

    题目大意就是有很多牛.告诉你每只牛的高度.然后有很多个询问.输出该区间内的最大身高差.也就是用RMQ求最大值最小值.貌似还可以用线段树.然而,我还不会线段树.....T_T 可能是太多组数据了.cin ...

  7. Find the hotel HDU - 3193(RMQ)

    题意: 有n个旅馆,从这n个旅馆中找出若干个旅馆,使得这若干个旅馆满足这样的条件:不能从其它和剩下的旅馆中找到一个价格和距离都小于这个旅馆的旅馆... 解析: 按price 排序,若price相同, ...

  8. hdoj5875【二分+RMQ】

    全部从我大哥那里学习得来.. 一开始硬着头皮就是根据思路上线段树,明知是T还要写(因为线段树还不是很熟,趁机练一发) 后来果然T了,然后就去学了一发RMQ的ST算法,查询是O(1). ST算法主要: ...

  9. 【转】最近公共祖先(LCA)

    基本概念 LCA:树上的最近公共祖先,对于有根树T的两个结点u.v,最近公共祖先LCA(T,u,v)表示一个结点x,满足x是u.v的祖先且x的深度尽可能大. RMQ:区间最小值查询问题.对于长度为n的 ...

随机推荐

  1. elementUI vue 页面加载的时候页面出现了黑字 页面优化处理 按钮弹出框文字

    elementUI 页面如果需要加载很多东西的时候, 自己定义的按钮或者弹出框dialog的文字就会显示在页面上, 一闪而过, 因此需要优化一下, elementUI 提供的loading有遮罩层, ...

  2. [总结]vue开发常见知识点及问题资料整理(持续更新)

    package.json中的dependencies与devDependencies之间的区别 –save-dev 和 –save 的区别 我们在使用npm install 安装模块或插件的时候,有两 ...

  3. spring boot 集成 redis lettuce

    一.简介 spring boot框架中已经集成了redis,在1.x.x的版本时默认使用的jedis客户端,现在是2.x.x版本默认使用的lettuce客户端,两种客户端的区别如下 # Jedis和L ...

  4. 关于处理注册表权限无法修改的问题(无法打开主键或注册表项unknown)

    CMD下(管理员) secedit /configure /cfg %windir%\inf\defltbase.inf /db defltbase.sdb /verbose 此命令可以生成报告, 任 ...

  5. loadrunner 脚本录制-Action分类

    脚本录制-Action分类 by:授客 QQ:1033553122 Action分类 l . Vuser_init 2. Vuser_end 3.  Action 在lr中用户的初始化操作应该存放在V ...

  6. Linux 之父自传《just for fun》读书笔记

    一次偶然的机会,看到了阮一峰老师关于这本书的介绍,当时我就觉得这本书相当有趣. 在没有读这本书之前,我觉得 linus 作为发明 Linux 系统的人,应该是一个比较严肃的人,就像我的老师一样.但事实 ...

  7. 接口的绑定方案和动态SQL

    1. 接口绑定方案 MyBatis中, 提供了一套接口绑定方案. 程序员可以提供一个接口, 然后提供对应接口的一个mapper.xml文件. MyBatis会自动将接口和xml文件进行绑定. 实际上就 ...

  8. python with open as f写中文乱码

    python3和python2的写法不一样具体如下: python3: with open(r'd:\ssss.txt','w',encoding='utf-8') as f: f.write(u'中 ...

  9. jQuery如何判断input元素是否获得焦点(点击编辑时)

    问题提出 如果你要判断input元素是否获得焦点,或者是否处在活动编辑状态,使用jQuery的 hasFocus() 方法或 is(':focus') 方法貌似都无效!搜索网上给出的办法,几乎净是采用 ...

  10. SQL SERVER查询字段在哪个表里

    ); SET @ColumnName='字段名的模糊匹配'; SELECT 表名=D.NAME, 表说明 THEN ISNULL(F.VALUE, ' ') ELSE ' ' END, 字段序号 = ...