zoukankan      html  css  js  c++  java
  • Crowd HDU

    通过坐标变换x'=x-y+n和y'=x+y,开大两倍空间就可以用一个两倍大的二维矩阵中

    从[x'-a,y'-a]到[x'+a,y'+a]这个子矩阵表示原图中到(x,y)曼哈顿距离小于等于a的菱形区域啦

    二维的离散化就是把(x,y)映射到x*n+y这个一维数列上

    这题极限空间应该是log(2n)*log(2n)*m吧。。这是2e7左右了呀,不知道4e6怎么过的?可能离散不出那么多严格的logN吧

     1 //#include<bits/stdc++.h>  
     2 //#pragma comment(linker, "/STACK:1024000000,1024000000")   
     3 #include<stdio.h>  
     4 #include<algorithm>  
     5 #include<queue>  
     6 #include<string.h>  
     7 #include<iostream>  
     8 #include<math.h>                    
     9 #include<stack>
    10 #include<set>  
    11 #include<map>  
    12 #include<vector>  
    13 #include<iomanip> 
    14 #include<bitset>
    15 using namespace std;         //
    16 
    17 #define ll long long  
    18 #define ull unsigned long long
    19 #define pb push_back  
    20 #define FOR(a) for(int i=1;i<=a;i++) 
    21 #define sqr(a) (a)*(a)
    22 #define dis(a,b) sqrt(sqr(a.x-b.x)+sqr(a.y-b.y))
    23 ll qp(ll a,ll b,ll mod){
    24     ll t=1;while(b){if(b&1)t=t*a%mod;b>>=1;a=a*a%mod;}return t;
    25 }
    26 struct DOT{int x;int y;};
    27 inline void read(int &x){int k=0;char f=1;char c=getchar();for(;!isdigit(c);c=getchar())if(c=='-')f=-1;for(;isdigit(c);c=getchar())k=k*10+c-'0';x=k*f;} 
    28 void ex(){puts("No");exit(0);}
    29 const int dx[4]={0,0,-1,1};
    30 const int dy[4]={1,-1,0,0};
    31 const int inf=0x3f3f3f3f; 
    32 const ll Linf=0x3f3f3f3f3f3f3f3fLL;
    33 const ll Mod=1e18+7;
    34 const double eps=1e-6;
    35 const double pi=acos(-1.0);
    36 
    37 
    38 
    39 inline int lowbit(int x){return x&-x;}
    40 
    41 const int maxn=8e4+33;
    42 
    43 int op[maxn],posx[maxn],posy[maxn];
    44 int v[maxn],h[4000050],tree[4000050],n,m,N,cnt;
    45 
    46 void add(int x,int y,int z){
    47     for(int i=x;i<=N;i+=lowbit(i)){
    48         for(int j=y;j<=N;j+=lowbit(j)){
    49             int pos=lower_bound(h,h+cnt,i*N+j)-h;
    50             tree[pos]+=z;
    51         }
    52     }
    53 }
    54 int ask(int x,int y){
    55     int ans=0;
    56     for(int i=x;i>0;i-=lowbit(i)){
    57         for(int j=y;j>0;j-=lowbit(j)){
    58             int pos=lower_bound(h,h+cnt,i*N+j)-h;
    59             if(h[pos]==i*N+j)ans+=tree[pos];
    60         }
    61     }
    62     return ans;
    63 }
    64 
    65 int main(){
    66     while(scanf("%d",&n)&&n){
    67         memset(tree,0,sizeof tree);
    68         int x,y;
    69         N=2*n;
    70         cnt=0;
    71         scanf("%d",&m);
    72         for(int i=1;i<=m;i++){
    73             scanf("%d%d%d%d",&op[i],&x,&y,&v[i]);
    74             int xx=x-y+n;int yy=x+y;
    75             posx[i]=xx;
    76             posy[i]=yy;
    77             if(op[i]==1){
    78                 for(int j=xx;j<=N;j+=lowbit(j)){
    79                     for(int k=yy;k<=N;k+=lowbit(k)){
    80                         h[cnt++]=j*N+k;
    81                     }
    82                 }
    83             }
    84         }
    85         sort(h,h+cnt);
    86         for(int i=1;i<=m;i++){
    87             if(op[i]==1)add(posx[i],posy[i],v[i]);
    88             else{
    89                 int X1=max(1,posx[i]-v[i]),Y1=max(1,posy[i]-v[i]);
    90                 int X2=min(N,posx[i]+v[i]),Y2=min(N,posy[i]+v[i]);
    91                 printf("%d
    ",ask(X2,Y2)+ask(X1-1,Y1-1)
    92                         -ask(X2,Y1-1)-ask(X1-1,Y2)
    93                         );
    94             }
    95         }
    96     }
    97 }
    View Code
  • 相关阅读:
    php生成二维码
    赞的算法
    Linux系统信息查看命令大全
    详细介绍Linux telnet命令的使用
    Linux VSFTP服务器
    禁止浏览器缓存页面的方法
    php开启短标签
    BZOJ2648 SJY摆棋子(KD-Tree)
    KD-Tree学习笔记
    BZOJ5461 PKUWC2018Minimax(概率期望+线段树合并+动态规划)
  • 原文地址:https://www.cnblogs.com/Drenight/p/8678516.html
Copyright © 2011-2022 走看看