RMQ ( 范围最小值查询 ) 问题是一种动态查询问题,它不需要修改元素,但要及时回答出数组 A 在区间 [l, r] 中最小的元素值。

RMQ(Range Minimum/Maximum Query):对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A中下标在i,j之间的最小/大值。

对于 RMQ ,我们通常关心两方面的算法效率:预处理时间和查询时间。
解决一般 RMQ 问题的三种方法
胜者树 (Winner Tree) O(n)-O(logn)
稀疏表 (Sparse Table) O(nlogn)-O(1)
线段树 (Segment Tree) O(n)-O(logn)
这里介绍一个稀疏表算法。

一、稀疏表ST算法--预处理

二、稀疏表ST算法--查询

三、RMQ ST模板

package ads;

public class RMQ_ST {

    int[] a;
int[][] st;//st[i][j],i表示数组a的索引为i的元素,即i:索引
int n;
public RMQ_ST(int[] a) {
this.a = a;
this.n = a.length;
st = new int[n][17];
} void initRMQ() {
int k = (int)(Math.log((double)n) / Math.log(2.0));
for (int i=0; i<n; i++) st[i][0] = i;
for (int j=1; j<=k; j++) { //dp开始,从1到k
for (int i=0; i<n; i++) { //遍历每个元素
st[i][j] = st[i][j-1]; //先赋值为前一半的RMQ值
int part = i + (1 << (j-1));//后一半的第一个元素索引
if (part >= n) break; //如果后一半的第一个元素已经超出数组范围,直接跳过后面的元素
if (a[ st[i][j-1] ] > a[ st[part][j-1] ]) st[i][j] = st[part][j-1];
}
}
} int rmq(int x, int y) {
int k = (int)(Math.log((double)y-x+1) / Math.log(2.0));
int part = y - (1<<k) + 1;//切分,比如0~4切分成st(0,2)和st(1,2),前者是0~3的最小元素,后者是1~4的
if (a[ st[x][k] ] < a[ st[part][k] ]) return st[x][k];
return st[part][k];
} public static void main(String[] args) {
int[] numbers = {4,3,2,1,6,7,8,9};
RMQ_ST st = new RMQ_ST(numbers);
st.initRMQ();
System.out.println(st.rmq(0, 3));
System.out.println(st.rmq(4, 7));
} }

四、示例代码

poj2425 http://poj.org/problem?id=2452

思路:枚举每个位置 i ,找出右边第一个比 a[i] 小的元素位置j

  在 i 到 j-1 中间求最大值的位置 k ,如果 a[k] > a[i],那么更新答案

Description

Xuanxuan has n sticks of different length. One day, she puts all her sticks in a line, represented by S1, S2, S3, ...Sn. After measuring the length of each stick Sk (1 <= k <= n), she finds that for some sticks Si and Sj (1<= i < j <= n), each stick placed between Si and Sj is longer than Si but shorter than Sj.

Now given the length of S1, S2, S3, …Sn, you are required to find the maximum value j - i.

Input

The input contains multiple test cases. Each case contains two lines. 
Line 1: a single integer n (n <= 50000), indicating the number of sticks. 
Line 2: n different positive integers (not larger than 100000), indicating the length of each stick in order.

Output

Output the maximum value j - i in a single line. If there is no such i and j, just output -1.

Sample Input

4
5 4 3 6
4
6 5 4 3

Sample Output

1
-1

Source

 
#include <iostream>
#include <cstdio>
#include <cmath> using namespace std; const int MAXN = ; int n;
int a[MAXN];
int st1[MAXN][], st2[MAXN][];//分别是最小RMQ和最大RMQ
int d[]; void rmqinit() {
for (int i=; i<=n; i++) st1[i][] = st2[i][] = i;
int k = int(log(double(n)) / log(2.0)) + ;
for (int j=; j<k; j++) {
for (int i=; i<n; i++) {
st1[i][j] = st1[i][j-];
st2[i][j] = st2[i][j-];
int part = i + ( << (j-));
if (part >= n) break;
if (a[ st1[i][j-] ] > a[ st1[part][j-] ]) st1[i][j] = st1[part][j-];
if (a[ st2[i][j-] ] < a[ st2[part][j-] ]) st2[i][j] = st2[part][j-];
}
}
} int rmqmin(int l, int r) {
int k = int( log(double(r-l+)) / log(2.0) );
int part = r - (<<k) + ;
if (a[ st1[l][k] ] < a[ st1[part][k] ]) return st1[l][k];
return st1[part][k];
} int rmqmax(int l, int r) {
int k = int( log(double(r-l+)) / log(2.0) );
int part = r - (<<k) + ;
if (a[ st2[l][k] ] > a[ st2[part][k] ]) return st2[l][k];
return st2[part][k];
} int bin_search(int a) {
int l = a, r = n - ;
while (l < r) {
int mid = ((l+r)>>) + ((l+r)&);
if (a == rmqmin(a, mid)) { //如果l~mid范围内的最小元素是l,说明l~mid都大于元素l
l = mid;
}
else {
r = mid - ;
}
}
return l;
} int work() {
int res = -;
for (int i=; i<n; i++) {
int r = rmqmax(i, bin_search(i));
res = max(res, r-i);
}
if (res == ) return -;
return res;
} int main()
{
while (scanf("%d", &n) != EOF) {
for (int i=; i<n; i++) {
scanf("%d", a+i);
} rmqinit();
printf("%d\n", work());
} //cout << "Hello world!" << endl;
return ;
}

RMQ (Range Minimal Query) 问题 ,稀疏表 ST的更多相关文章

  1. 算法学习 - ST表 - 稀疏表 - 解决RMQ问题

    2017-08-26 21:44:45 writer:pprp RMQ问题就是区间最大最小值查询问题: 这个SparseTable算法构造一个表,F[i][j] 表示 区间[i, i + 2 ^ j ...

  2. 基于稀疏表(Sparse Table)的RMQ(区间最值问题)

    在RMQ的其他实现方法中,有一种叫做ST的算法比较常见. [构建] dp[i][j]表示的是从i起连续的2j个数xi,xi+1,xi+2,...xi+2j-1( 区间为[i,i+2j-1] )的最值. ...

  3. 动态规划——稀疏表求解RMQ问题

    RMQ (Range Minimum/Maximum Query)问题,即区间最值查询问题,是求解序列中的某一段的最值的问题.如果只需要询问一次,那遍历枚举(复杂度O(n))就是最方便且高效的方法,但 ...

  4. ST (Sparse Table:稀疏表)算法

    1541:[例 1]数列区间最大值 时间限制: 1000 ms         内存限制: 524288 KB提交数: 600     通过数: 207 [题目描述] 输入一串数字,给你 MM 个询问 ...

  5. AOJ DSL_2_A Range Minimum Query (RMQ)

    Range Minimum Query (RMQ) Write a program which manipulates a sequence A = {a0,a1,...,an−1} with the ...

  6. Range Minimum Query and Lowest Common Ancestor

    作者:danielp 出处:http://community.topcoder.com/tc?module=Static&d1=tutorials&d2=lowestCommonAnc ...

  7. AOJ DSL_2_E Range Add Query (RAQ)

    Range Add Query 数列 A = {a1,a2,...,an} に対し.次の2つの操作を行うプログラムを作成せよ. add(s,t,x): as,as+1,...,at にxを加算する. ...

  8. AOJ DSL_2_D Range Update Query (RUQ)

    Range Update Query 数列 A = {a0,a1 ,...,an−1} に対し.次の2つの操作を行うプログラムを作成せよ. update(s,t,x): as,as+1,...,at  ...

  9. [LeetCode] Range Sum Query 2D - Mutable 二维区域和检索 - 可变

    Given a 2D matrix matrix, find the sum of the elements inside the rectangle defined by its upper lef ...

随机推荐

  1. hiho42 : 骨牌覆盖问题·二

    描述 上一周我们研究了2xN的骨牌问题,这一周我们不妨加大一下难度,研究一下3xN的骨牌问题?所以我们的题目是:对于3xN的棋盘,使用1x2的骨牌去覆盖一共有多少种不同的覆盖方法呢?首先我们可以肯定, ...

  2. arch框架人员、组织说明

    目前ERP辅助系统集成了三大模块功能,分别是财务辅助.物理辅助.报账平台. 财务辅助模块人员在ARCH_USER 表中进行管理,通过单独的[用户映射功能]将ARCH系统用户和ERP用户进行关联,关联信 ...

  3. lisp分支

    newLISP http://www.ituring.com.cn/article/110968 clojure             http://clojure.org/             ...

  4. HW 研发体系机构的几个术语

    PDT(product development team)产品开发团队   类似于产品经理 程序员 --  PL -- PM  --开发代表 -- PDT LEADER --------------- ...

  5. Bluetooth GAP介绍

    目录 1 GAP协议栈 2 Profile Role 3 用户接口 4 模式 5 安全 5.1 认证(Authentication) 5.2 安全模式 6 Idle Mode Procedures 7 ...

  6. 关于网站的UV分析

    一:准备 1.统计的维度 guid tracktime provice 2.key与value的设定 key:date+provice_guid value:NullWritable 3.案例分析 表 ...

  7. hadoop-2.7.3 在windows环境下安装(无需Cygwin)

    http://blog.csdn.net/kokjuis/article/details/53537029

  8. JS:checkFrom对输入框和文本框的判断总结

    天看了老东家的一个专题页面,发现里边的checkFrome.js收集了很多对文本框的判断,非常有用收藏一下.其中包含了:1.页面截取字符串2.文本框最大长度限制3.判断必须是数字和字母的组合4.判断是 ...

  9. 其他常用HTML 片段

    1.input placeholder 文字居中 字体大小+上下padding值等于设计稿宽度 设计稿中总高度为86px   padding:27px 0;font-size:30px;   2.英文 ...

  10. ArcGIS API for Silverlight 当DataGrid选中项时,地图聚焦弹出窗口,并可以播放音频文件

    原文:ArcGIS API for Silverlight 当DataGrid选中项时,地图聚焦弹出窗口,并可以播放音频文件 先看效果图,然后上代码: <UserControl x:Class= ...