zoukankan      html  css  js  c++  java
  • rmq学习

    参考博客:https://blog.csdn.net/qq_31759205/article/details/75008659

    即区间最值查询,是指这样一个问题:对于长度为n的数列A,回答若干次询问RMQ(i,j),返回数列A中下标在区间[i,j]中的最小/大值。

    本文介绍一种比较高效的ST算法解决这个问题。ST(Sparse Table)算法可以在O(nlogn)时间内进行预处理,然后在O(1)时间内回答每个查询。

    //含义:算法分析:这个算法就是基于DP和位运算符,
    //我们用dp【i】【j】表示从第 i 位开始,到第 i + 2^j -1 位的最大值或者最小值
    
    /*
    求解:  dp【i】【j】的时候可以把它分成两部分,
    第一部分从 i 到 i + 2 ^( j-1 ) - 1 ,
    第二部分从 i + 2 ^( j-1 )  到 i + 2^j - 1 次方,
    其实我们知道二进制数后一个是前一个的二倍,那么可以把 i ---  i + 2^j
    这个区间 通过2^(j-1) 分成相等的两部分, 那么转移方程很容易就写出来了
    mm[i] [j] = max ( mm [ i ] [ j - 1 ] , mm [ i + ( 1 << ( j - 1 ) ) ] [ j - 1 ] );
    */
    void make_rmq(){
        for(int i=1;i<=n;++i)
            dpmax[i][0]=dpmin[i][0] = a[i];
        for(int j=1;(1<<j)<=n;j++){
            for(int i=1;i+(1<<j)-1<=n;i++){
                dpmax[i][j]= max(dpmax[i][j-1],dpmax[i+(1<<(j-1))][j-1]);
                dpmin[i][j] = min(dpmin[i][j-1],dpmin[i+(1<<(j-1))][j-1]);
            }
        }
    }
    
    /*
    查询:
    
    查询的时候对于任意一个区间 l -- r ,
    我们同样可以得到区间差值 len = (r - l + 1)。
    那么我们这一用小于2^k<=len,的 k 把区间
    分成可以交叉的两部分l 到 l+2^(k)- 1,
    到 r -(1<<k)+1 到 r 的两部分,很easy的求解了。
    */
    int rmq(int l,int r){
    
        int len =r-l+1;
        int k=0;
        while((1<<(k+1))<=len)
            ++k;
        //得到的区间会有部分重叠 但是不会超过l,r
        //l+2^(k)-1
        //r  +2^k-1
        int ans1= max(dpmax[l][k],dpmax[r-(1<<k)+1][k]);
        int ans2= min(dpmin[l][k],dpmin[r-(1<<k)+1][k]);
        return ans1-ans2;
    }
  • 相关阅读:
    C#呓语
    引起超时的原因及表解锁的方法<转>
    如何使用数据库引擎优化顾问优化数据库 <转>
    缩短IIS应用池回收时间,减少IIS假死<转>
    Microsoft Silverlight 4 Tools for Visual Studio 2010中文版本
    系统统一验证(IHttpHandlerFactory)<转>
    解决CSS BUG的顺口溜<转>
    重建索引提高SQL Server性能<转>
    .NET调用osql.exe执行sql脚本创建表和存储过程<转>
    SQL SERVER性能优化综述<转>
  • 原文地址:https://www.cnblogs.com/wjhstudy/p/9836134.html
Copyright © 2011-2022 走看看