题解 P6290 【[eJOI2017]粒子】
AThousandSuns · · 题解
每次操作时二分,判断此时跑到最右的
由于我相信
其实就是求个半平面交的事。每次重构半平面交,然后把两边的上半部分分段加起来,在上面判断。
由于直线就那么几条,所以不用每次排序。
时间复杂度
然后当我发现评测记录里一大堆 0.9s+ 的时候心态就崩了。
第二天我过来补代码,补着补着发现时限甚至变成了 2s,然后提交记录前几个 1.8s+。看起来就是暴力常数太大然后申请开大时限。
然后开始无 能 狂 怒。是真的无能,由于一些原因当时发不了犇犇和帖子,连博客都不能发。
然而改时限其实很没道理,因为官方题解也说了 所以我就只能干难受了
拿了个最优解不亏
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int> PII;
const int maxn=50050,mod=998244353;
#define MP make_pair
#define PB push_back
#define lson o<<1,l,mid
#define rson o<<1|1,mid+1,r
#define FOR(i,a,b) for(int i=(a);i<=(b);i++)
#define ROF(i,a,b) for(int i=(a);i>=(b);i--)
#define MEM(x,v) memset(x,v,sizeof(x))
inline ll read(){
char ch=getchar();ll x=0,f=0;
while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return f?-x:x;
}
struct line{
int k,id1,id2;
ll b;
bool operator<(const line &l)const{
if(k!=l.k) return k<l.k;
return b>l.b;
}
}lx[maxn],ly[maxn],hx[maxn],hy[maxn],h[maxn*2];
int n,l,k,tx[maxn],vx[maxn],ty[maxn],vy[maxn],tpx,tpy,tot;
inline double interx(line l1,line l2){
return 1.0*(l1.b-l2.b)/(l2.k-l1.k);
}
line merge(line l1,line l2){
return (line){l1.k+l2.k,l1.id1,l2.id1,l1.b+l2.b};
}
int main(){
n=read();l=read();k=read();
FOR(i,1,n) tx[i]=read(),vx[i]=read(),lx[i]=(line){vx[i],i,0,-1ll*vx[i]*tx[i]};
FOR(i,1,n) ty[i]=read(),vy[i]=read(),ly[i]=(line){vy[i],i,0,-1ll*vy[i]*ty[i]};
sort(lx+1,lx+n+1);sort(ly+1,ly+n+1);
while(k--){
tpx=0;
FOR(i,1,n){
if(tpx && lx[i].k==hx[tpx].k) continue;
while(tpx>=2 && interx(lx[i],hx[tpx])<=interx(lx[i],hx[tpx-1])) tpx--;
hx[++tpx]=lx[i];
}
tpy=0;
FOR(i,1,n){
if(tpy && ly[i].k==hy[tpy].k) continue;
while(tpy>=2 && interx(ly[i],hy[tpy])<=interx(ly[i],hy[tpy-1])) tpy--;
hy[++tpy]=ly[i];
}
tot=0;
int j=1;
FOR(i,1,tpx){
double lft=i==1?-1e18:interx(hx[i],hx[i-1]),rig=i==tpx?1e18:interx(hx[i],hx[i+1]);
while(j<=tpy){
double L=j==1?-1e18:interx(hy[j],hy[j-1]),R=j==tpy?1e18:interx(hy[j],hy[j+1]);
if(L>rig) break;
if(R>=lft) h[++tot]=merge(hx[i],hy[j]);
if(R>rig) break;
j++;
}
}
int ans1=0,ans2=0;
FOR(i,1,tot){
double lft=i==1?-1e18:interx(h[i],h[i-1]),rig=i==tot?1e18:interx(h[i],h[i+1]);
double x=1.0*(l-h[i].b)/h[i].k;
if(x>=lft && x<=rig){
ans1=h[i].id1;
ans2=h[i].id2;
break;
}
}
printf("%d %d\n",ans1,ans2);
FOR(i,1,n-1) if(lx[i].id1==ans1) swap(lx[i],lx[i+1]);
FOR(i,1,n-1) if(ly[i].id1==ans2) swap(ly[i],ly[i+1]);
n--;
}
}