1. 程式人生 > >HDU 1394 Minimum Inversion Number 線段樹

HDU 1394 Minimum Inversion Number 線段樹

acm log 我們 oid query sca view ace 吃飯

  題目鏈接: http://acm.hdu.edu.cn/showproblem.php?pid=1394

  題目大意:給出一個n排列, 進行n中變換, 求最大逆序數

  解題思路: 已知一個數列的逆序數, 那麽他的變換可以O(1)求出來, 所以現在的主要問題就是求一組排列的逆序數, n 小於等於5000, 完全可以暴力, 但是現在用線段樹做。由於是一組n排列, 而且我們只關心一個數前面的比他大的數, 那當我們插入a[i]時, 只需要查詢0 ~ i-1 中 a[i] ~ n-1 內的數的個數就可以了, 我們可以先造一顆0的空樹, 這樣我們查詢的時候就就直接查全部序列就可以。(吃飯去了)

  代碼:

技術分享
#include <iostream>
#include 
<cstdio> #include <string> #include <vector> #include <map> #include <cstring> #include <iterator> #include <cmath> #include <algorithm> #include <stack> #include <deque> #include <map> #define lson l, m, rt<<1 #define rson m+1, r, rt<<1|1 const
int maxn = 5000+7; using namespace std; int sum[4*maxn]; int arr[maxn]; void Pushplus( int rt ) { sum[rt] = sum[rt<<1] + sum[rt<<1|1]; return; } void build( int l, int r, int rt ) { sum[rt] = 0; if( l == r ) return; int m = (l + r) >> 1; build( lson ); build( rson );
return; } int Query( int L, int R, int l, int r, int rt ) { if( L <= l && r <= R ) { return sum[rt]; } int m = (l + r) >> 1; int ret = 0; if( L <= m ) ret += Query( L, R, lson ); if( R > m ) ret += Query( L, R, rson ); return ret; } void update(int p, int l, int r, int rt) { if( l == r ) { sum[rt]++; return; } int m = (l+r) >> 1; if( p <= m ) update(p, lson); else update(p, rson); Pushplus(rt); } int main() { int n; while( scanf( "%d", &n ) == 1 ) { build( 0, n-1, 1 ); int sum = 0; for( int i = 0; i < n; i++ ) { scanf( "%d", arr + i ); sum += Query( arr[i], n-1, 0, n-1, 1 ); update(arr[i], 0, n-1, 1); } int ret = sum; for( int i = 0; i < n; i++ ) { sum += n - arr[i] - arr[i] - 1; ret = min( ret, sum ); } printf( "%d\n", ret ); } }
View Code

  思考:

HDU 1394 Minimum Inversion Number 線段樹