BZOJ1807 : [Ioi2007]Pairs 彼此能听得见的动物对数

一维的情况:

排序后维护一个单调指针即可,时间复杂度$O(n\log n)$。

二维的情况:

旋转坐标系后转化为二维数点问题,扫描线+树状数组维护即可,时间复杂度$O(n\log n)$。

三维的情况:

将后两维旋转坐标系,对于每个x,预处理出横坐标为x的点的后两维的二维前缀和。

枚举一个点,再枚举另一个点的x,在相应坐标的二维前缀和里查询正方形内点的个数即可,时间复杂度$O(nm)$。

 

#include<cstdio>
#include<algorithm>
#define N 300010
using namespace std;
int n,m,i,D;long long ans;
inline void read(int&a){char c;while(!(((c=getchar())>='0')&&(c<='9')));a=c-'0';while(((c=getchar())>='0')&&(c<='9'))(a*=10)+=c-'0';}
namespace task1{
int a[N],i,j;
void solve(){
  for(i=1;i<=n;i++)read(a[i]);
  sort(a+1,a+n+1);
  for(i=j=1;i<=n;ans+=i-j,i++)while(a[i]-a[j]>D)j++;
}
}
namespace task2{
int m,k,i,x,y,z,bit[N],q[N],X,Y,b[N];
struct P{int x,y,z,t;P(){}P(int _x,int _y,int _z,int _t){x=_x,y=_y,z=_z,t=_t;}}a[N];
inline bool cmp(const P&a,const P&b){return a.x==b.x?a.t<b.t:a.x<b.x;}
inline int lower(int x){
  int l=1,r=m,t,mid;
  while(l<=r)if(b[mid=(l+r)>>1]<=x)l=(t=mid)+1;else r=mid-1;
  return t;
}
inline void add(int x,int y){for(;x<=m;x+=x&-x)bit[x]+=y;}
inline int ask(int x){int t=0;for(;x;x-=x&-x)t+=bit[x];return t;}
void solve(){
  for(m=0,z=D,i=1;i<=n;i++){
    read(x),read(y),X=x+y,Y=x-y;
    a[++m]=P(X,Y,0,i),b[m]=Y;
    a[++m]=P(X+z,Y-z,Y+z,n+1),b[m]=Y-z;
    a[++m]=P(X-z,Y-z,Y+z,0),b[m]=Y+z;
  }
  for(sort(a+1,a+m+1,cmp),sort(b+1,b+m+1),i=1;i<=m;i++){
    if(a[i].t&&a[i].t<=n)ans+=ask(lower(a[i].y));
    else{
      a[i].y=lower(a[i].y),a[i].z=lower(a[i].z)+1;
      if(a[i].t)add(a[i].y,-1),add(a[i].z,1);else add(a[i].y,1),add(a[i].z,-1);
    }
  }
  ans=(ans-n)/2;
}
}
namespace task3{
int i,j,k,x,y,z,X,Y,a[N][3],s[76][152][152];
inline int ask(int x,int y){return s[j][max(min(x,150),0)][max(min(y,150),0)];}
void solve(){
  for(i=1;i<=n;i++){
    read(x),read(y),read(z),X=y+z,Y=y-z+75;
    a[i][0]=x,a[i][1]=X,a[i][2]=Y;
    s[x][X][Y]++;
  }
  for(i=1;i<=75;i++)for(j=1;j<=150;j++)for(k=1;k<=150;k++)s[i][j][k]+=s[i][j-1][k]+s[i][j][k-1]-s[i][j-1][k-1];
  for(i=1;i<=n;i++)for(j=1;j<=75;j++)if(abs(a[i][0]-j)<=D){
    x=D-abs(a[i][0]-j);
    ans+=ask(a[i][1]+x,a[i][2]+x)-ask(a[i][1]-x-1,a[i][2]+x)-ask(a[i][1]+x,a[i][2]-x-1)+ask(a[i][1]-x-1,a[i][2]-x-1);
  }
  ans=(ans-n)/2;
}
}
int main(){
  read(m),read(n),read(D),read(i);
  if(m==1)task1::solve();
  if(m==2)task2::solve();
  if(m==3)task3::solve();
  return printf("%lld",ans),0;
}

  

posted @ 2015-10-07 00:45  Claris  阅读(651)  评论(0编辑  收藏  举报