光線追蹤node
30分暴力就是看每一個光線和每一個矩形的下邊界和左邊界交點c++
發現詢問很難,ide
離線下來操做,ui
能夠把角度離散化,轉化爲區間賦值,單點查詢問題spa
每次就是給屬於下邊界角度內賦max,左邊界角度內賦max指針
我忽然感受沒什麼可寫的code
#include<bits/stdc++.h> using namespace std; #define ll long long #define A 5471745 struct node{ ll ord,x; node(){} node(const ll &a,const ll &b){ord=a,x=b;} }; node Min(node a,node b){ return a.x<b.x?a:a.x==b.x?(a.ord<b.ord?b:a):b; } ll pan(ll x){ if(x>1e9) return 0; return x; } struct tree{ node w[A]; ll cl[A],cr[A]; void built(ll x,ll l,ll r){ cl[x]=l,cr[x]=r; w[x]=node(1e9+7,1e9+7); if(l==r){ return ; } ll mid=(l+r)>>1; built(x<<1,l,mid); built(x<<1|1,mid+1,r); } void change(ll x,ll l,ll r,ll ord,ll val){ if(cl[x]>=l&&cr[x]<=r){ // printf("l=%lld r=%lld w[].val=%lld val=%lld\n",cl[x],cr[x],w[x].x,val); w[x]=Min(w[x],node(ord,val)); return ; } ll mid=(cl[x]+cr[x])>>1; if(mid>=l) change(x<<1,l,r,ord,val); if(mid<r) change(x<<1|1,l,r,ord,val); } node ask(ll x,ll pla){ if(cl[x]==cr[x]) return w[x]; node p; ll mid=(cl[x]+cr[x])>>1; if(mid>=pla) p=ask(x<<1,pla); else p=ask(x<<1|1,pla); return Min(p,w[x]);//標記永久化 } }X,Y; ll x[A],xx[A],y[A],yy[A],opt[A]; ll mnx0=1e9+7,mnxid,mny0=1e9+7,mnyid,cnt,q; long double lsh[A]; long double cl(ll x){ if(x) return 1.0*x; return 1e-7L; } int main(){ scanf("%lld",&q); for(ll i=1;i<=q;i++){ scanf("%lld",&opt[i]); if(opt[i]==1){ scanf("%lld%lld%lld%lld",&x[i],&y[i],&xx[i],&yy[i]); lsh[++cnt]=1.0L*y[i]/cl(x[i]),lsh[++cnt]=1.0L*yy[i]/cl(x[i]),lsh[++cnt]=1.0L*y[i]/cl(xx[i]); } else { scanf("%lld%lld",&x[i],&y[i]); if(x[i]) lsh[++cnt]=1.0L*y[i]/x[i]; } } sort(lsh+1,lsh+cnt+1); ll len=unique(lsh+1,lsh+cnt+1)-lsh-1; X.built(1,1,len),Y.built(1,1,len); for(ll i=1;i<=q;i++){ if(opt[i]==1){ if(!x[i])if(mnx0>=y[i])mnx0=y[i],mnxid=i; if(!y[i])if(mny0>=x[i])mny0=x[i],mnyid=i; ll x1=lower_bound(lsh+1,lsh+len+1,1.0L*y[i]/cl(x[i]))-lsh; ll x3=lower_bound(lsh+1,lsh+len+1,1.0L*y[i]/cl(xx[i]))-lsh; ll x2=lower_bound(lsh+1,lsh+len+1,1.0L*yy[i]/cl(x[i]))-lsh; // printf("i=%lld\n",i); X.change(1,x1,x2,i,x[i]),Y.change(1,x3,x1,i,y[i]); } else { if(!x[i]){printf("%lld\n",mnxid);continue;} if(!y[i]){printf("%lld\n",mnyid);continue;} // printf("i=%lld lower_bound(lsh+1,lsh+1+len,1.0L*y[i]/x[i])-lsh=%lld\n",i,1ll*(lower_bound(lsh+1,lsh+1+len,1.0L*y[i]/x[i])-lsh)); node a1=X.ask(1,lower_bound(lsh+1,lsh+1+len,1.0L*y[i]/x[i])-lsh), a2=Y.ask(1,lower_bound(lsh+1,lsh+1+len,1.0L*y[i]/x[i])-lsh); ll fx=a1.x,fy=a2.x; // printf("fx=%lld fy=%lld\n",fx,fy); if(fx*y[i]==fy*x[i])printf("%lld\n",pan(max(a1.ord,a2.ord))); else if(fx*y[i]<fy*x[i])printf("%lld\n",pan(a1.ord)); else printf("%lld\n",pan(a2.ord)); } } }
臨面合併blog
奇怪的數據範圍應該給了必定的提示,8之內就基本是明示狀壓了it
狀態定義挺神的二進制下有一個1表明以當前點爲右端點向左延伸出一個矩形event
例如(假設原圖是1 1 1 1)那麼當前是0 1 0 1 表示兩個矩形
判斷一下是否能夠與上面合併,減去相同貢獻
過程能夠預處理
不要用單調指針處理,單調指針細節特別多,
預處理
void fg(ll x){ for(ll i=0;i<=maxn;i++){ if((i&least[x])!=least[x]) continue ; if((i|maxx[x])!=maxx[x]) continue ; belong[x].push_back(i); ll cnt=0; for(ll j=1;j<=m;j++){ if((i>>(j-1))&1) dl[++cnt]=j; } for(ll j=1;j<=cnt;j++){ if(j==1) v[x][i].push_back(base[dl[1]]&a[x]); else v[x][i].push_back((base[dl[j]]^base[dl[j-1]])&a[x]); } } }
減去相同貢獻
ll cnt=v[i][now].size(); for(ll j1=0;j1<v[i][now].size();j1++) for(ll j2=0;j2<v[i-1][last].size();j2++){ ll tox=v[i][now][j1],toy=v[i-1][last][j2]; if(tox==toy){ // printf("tox=%lld toy=%lld\n",tox,toy); cnt--; } }
出題人題解