poj 3067

题目:http://poj.org/problem?id=3067


树状数组,先对于左边的数从大到小,相同的把右边的也从大到小。。然后。。简单统计即可。

#include <cmath>
#include <ctime>
#include <iostream>
#include <string>
#include <vector>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <queue>
#include <map>
#include <set>
#include <algorithm>
#include <cctype>
#include <stack>
#include <deque>
using namespace std;
typedef long long LL;
#define EPS 10e-9
#define INF 0x3f3f3f3f
#define REP(i,n) for(int i=0; i<(n); i++)
const int maxn =1100000;
struct node{
    int a,b;
    bool operator <(const node &v) const{
         if(a==v.a) return b>v.b;
         return a>v.a;
    }
}a[maxn];
int c[maxn];
int lowbit(int x){ return x&-x;}
int add(int x){
    while(x<maxn){
        c[x]+=1;    x+=lowbit(x);
    }
}
LL sum(int x){
    LL ret=0;
    while(x>0){
        ret+=c[x];  x-=lowbit(x);
    }
    return ret;
}
int main(){
    int t,n,m,k,ca=1;
    scanf("%d",&t);
    while(t--){
        memset(c,0,sizeof(c));
        scanf("%d %d %d",&n,&m,&k);
        for(int i=0;i<k;i++){
            scanf("%d %d",&a[i].a,&a[i].b);
        }
        sort(a,a+k);
        LL ans=0;
        for(int i=0;i<k;i++){
            ans+=sum(a[i].b-1);
            add(a[i].b);

        }

        printf("Test case %d: %lld\n",ca++,ans);
    }



   return 0;
}



你可能感兴趣的:(poj 3067)