BZOJ3289 Mato的文件管理

好煩的題,作了我2h,看來蒟蒻就是弱啊。。。算法

感受就是莫隊算法來着,而後就走上了亂搞的不歸路、、、數組

莫隊算法詳情請上百度搜索,謝謝!>.<ide

這道題要求逆序對,因此能夠用樹狀數組動態維護。每次轉移複雜度就是樹狀數組點修改的複雜度,可是轉移的時候好煩。。。(必定是我太弱了)ui

再加上分塊,總複雜度爲O(n * sqrt(n) * log(n)),剛恰好卡過去。。。spa

 

  1 /**************************************************************
  2     Problem: 3289
  3     User: rausen
  4     Language: C++
  5     Result: Accepted
  6     Time:8772 ms
  7     Memory:10200 kb
  8 ****************************************************************/
  9  
 10 #include <cstdio>
 11 #include <cmath>
 12 #include <algorithm>
 13  
 14 using namespace std;
 15 typedef long long LL;
 16  
 17 struct data{
 18     int l, r, wl, w;
 19     LL ans;
 20 } a[150000];
 21 inline bool cmp(data a, data b){
 22     return a.wl == b.wl ? a.r < b.r : a.l < b.l;
 23 }
 24  
 25 inline bool cmp_id(data a, data b){
 26     return a.w < b.w;
 27 }
 28  
 29 LL x[150000], y[150000];
 30 LL BIT[150000], rank[150000];
 31 int TOT, T, tot, w[150000], block[150000];
 32 int n, Q, l, r, num;
 33 LL sum;
 34  
 35 void build(){   
 36     T = (int) sqrt(n), tot = 0;
 37     int x = 0;
 38     while (x < n)
 39         x += T, w[++tot] = x;
 40     w[tot] = min(w[tot], n);
 41     int y = 1;
 42     for (int i = 1; i <= n; ++i)
 43         if (i <= w[y]) block[i] = y;
 44         else block[i] = ++y;
 45 }
 46  
 47 inline int lowbit(int x){
 48     return x & (-x);
 49 }
 50  
 51 LL query(int x){
 52     if (x < 0) return 0;
 53     LL res = 0;
 54     while (x)
 55         res += BIT[x], x -= lowbit(x);
 56     return res;
 57 }
 58  
 59 void change(int x, int del){
 60     while (x <= n)
 61         BIT[x] += del, x += lowbit(x);
 62 }
 63  
 64 void update_right(int p, int del){
 65     sum += (LL) del * (num - query(x[p])), num += del;
 66     change(x[p], del);
 67 }
 68  
 69 void update_left(int p, int del){
 70     change(x[p], del);  
 71     sum += (LL) del * query(x[p] - 1), num += del;
 72 }
 73  
 74 int find(int x){
 75     int l = 1, r = TOT, mid;
 76     for (; l != r; ){
 77         mid = (l + r) >> 1;
 78         if (rank[mid] >= x) r = mid;
 79         else l = mid + 1;
 80     }
 81     return l;
 82 }
 83  
 84 int main(){
 85     scanf("%d", &n);
 86     for (int i = 1; i <= n; ++i){
 87         scanf("%lld", x + i);
 88         y[i] = x[i];
 89     }
 90     build();
 91     sort(y + 1, y + n + 1);
 92     rank[++TOT] = y[1];
 93     for (int i = 2; i <= n; ++i)
 94         if (y[i] != y[i - 1])
 95             rank[++TOT] = y[i];
 96     for (int i = 1; i <= n; ++i)
 97         x[i] = find(x[i]);
 98      
 99     scanf("%d", &Q);
100     for (int i = 1; i <= Q; ++i){
101         scanf("%d%d", &a[i].l, &a[i].r);
102         a[i].wl = block[a[i].l], a[i].w = i;
103     }
104     sort(a + 1, a + Q + 1, cmp);
105     l = 1, r = 0, num = sum = 0;
106     for (int i = 1; i <= Q; ++i){
107         for (; r < a[i].r; ++r) update_right(r + 1, 1);
108         for (; r > a[i].r; --r) update_right(r, -1);
109         for (; l < a[i].l; ++l) update_left(l, -1);
110         for (; l > a[i].l; --l) update_left(l - 1, 1);
111         a[i].ans =  a[i].l == a[i].r ? 0 : sum;
112     }
113      
114     sort(a + 1, a + Q + 1, cmp_id);
115     for (int i = 1; i <= Q; ++i)
116         printf("%lld\n", a[i].ans);
117     return 0;
118 }
View Code
相關文章
相關標籤/搜索