題目鏈接 給出一個有N個數的序列,編號0 - N - 1。進行Q次查詢,查詢編號i至j的所有數中,最大的數是多少。 例如: 1 7 6 3 1。i = 1, j = 3,對應的數為7 6 3,最大的數為7。(該問題也被稱為RMQ問題) 輸入 輸出 輸入樣例 輸出樣例 Sparse Table解決Ra ...
給出一個有N個數的序列,編號0 - N - 1。進行Q次查詢,查詢編號i至j的所有數中,最大的數是多少。
例如: 1 7 6 3 1。i = 1, j = 3,對應的數為7 6 3,最大的數為7。(該問題也被稱為RMQ問題)輸入
第1行:1個數N,表示序列的長度。(2 <= N <= 10000) 第2 - N + 1行:每行1個數,對應序列中的元素。(0 <= S[i] <= 10^9) 第N + 2行:1個數Q,表示查詢的數量。(2 <= Q <= 10000) 第N + 3 - N + Q + 2行:每行2個數,對應查詢的起始編號i和結束編號j。(0 <= i <= j <= N - 1)
輸出
共Q行,對應每一個查詢區間的最大值。
輸入樣例
5 1 7 6 3 1 3 0 1 1 3 3 4
輸出樣例
7 7 3
Sparse Table解決Range Minimum/Maximum Query學習參考博客
分析
ST用DP O(nlogn)預處理 ,O(1)查詢。
設a[i]是要求區間最值的數列,RMQ[i, j]表示從第i個數起連續2^j個數中的最大值。
例如:
a數列為:3 2 4 5 6 8 1 2 9 7
RMQ[1,0]表示第1個數起,長度為2^0=1的最大值,其實就是3這個數。同理 RMQ[1,1] = max(3,2) = 3, RMQ[1,2]=max(3,2,4,5) = 5,RMQ[1,3] = max(3,2,4,5,6,8,1,2) = 8;
且[i,0]就等於a[i]。
狀態轉移方程RMQ[i, j]=max(RMQ[i,j-1], RMQ[i + 2^(j-1),j-1])。
![](https://images.cnblogs.com/OutliningIndicators/ContractedBlock.gif)
1 #include <iostream> 2 #include <cstdio> 3 #include <cmath> 4 #include <cstring> 5 #define maxn 10005 6 using namespace std; 7 8 int a[maxn],RMQ[maxn][15]; 9 10 void RMQ_INIT(int n){ 11 for(int i = 0; i < n; ++i) RMQ[i][0] = a[i]; 12 for(int j = 1; (1<<j) <= n; ++j) 13 for(int i = 0; i + (1<<j) - 1 < n; ++i) 14 RMQ[i][j] = max(RMQ[i][j-1], RMQ[i + (1<<(j-1))][j-1]); 15 } 16 17 int Find(int L, int R){ 18 int k = 0; 19 while((1<<(k+1)) <= R-L+1) k++; 20 return max(RMQ[L][k], RMQ[R-(1<<k)+1][k]); 21 } 22 23 int main(){ 24 int N,Q,L,R; 25 while(cin>>N){ 26 memset(RMQ, 0, sizeof(RMQ)); 27 for(int i=0;i<N;++i) cin>>a[i]; 28 RMQ_INIT(N); 29 cin>>Q; 30 while(Q--){ 31 cin>>L>>R; 32 cout<<Find(L,R)<<endl; 33 } 34 } 35 return 0; 36 }View Code