这里有一辆赛车比赛正在进行,赛场上一共有N辆车,分别称为个g1,g2……gn。赛道是一条无限长的直线。最初,gi位于距离起跑线前进ki的位置。比赛开始后,车辆gi将会以vi单位每秒的恒定速度行驶。在这个比赛过程中,如果一辆赛车曾经处于领跑位置的话(即没有其他的赛车跑在他的前面),这辆赛车最后就可以得奖,而且比赛过程中不用担心相撞的问题。现在给出所有赛车的起始位置和速度,你的任务就是算出那些赛车将会得奖。
对于100%的数据N<=10000, 0<=ki<=10^9, 0<=vi<=10^9
2016.1.17新加数据一组 By Nano_ape
计算几何,维护一个y轴右面的半平面交。
注意两个赛车速度和起始状态都相同的情况。
#include<iostream> #include<cstdio> #include<cstring> #include<cmath> #include<cstdlib> #include<algorithm> #include<vector> #define F(i,j,n) for(int i=j;i<=n;i++) #define D(i,j,n) for(int i=j;i>=n;i--) #define ll long long #define maxn 10005 #define eps 1e-8 using namespace std; int n,top,ans[maxn],f[maxn]; struct car{int p,v,id;}a[maxn],s[maxn]; vector<int> v[maxn]; inline int read() { int x=0,f=1;char ch=getchar(); while (ch<'0'||ch>'9'){if (ch=='-') f=-1;ch=getchar();} while (ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();} return x*f; } inline bool cmp(car x,car y) { return (x.v==y.v)?x.p>y.p:x.v<y.v; } inline double cross(car x,car y) { return (double)(x.p-y.p)/(y.v-x.v); } int main() { n=read(); F(i,1,n) a[i].p=read(); F(i,1,n) a[i].v=read(); F(i,1,n) a[i].id=i,f[i]=i; sort(a+1,a+n+1,cmp); s[++top]=a[1]; F(i,2,n) { if (a[i].v==a[i-1].v) { if (a[i].p==a[i-1].p) { v[f[a[i-1].id]].push_back(a[i].id); f[a[i].id]=f[a[i-1].id]; } continue; } while (top>=2&&cross(s[top],s[top-1])>cross(s[top-1],a[i])) top--; while (top>=1&&cross(s[top],a[i])<-eps) top--; s[++top]=a[i]; } F(i,1,top) ans[i]=s[i].id; int tt=top; F(i,1,tt) for(int j=0;j<v[ans[i]].size();j++) ans[++top]=v[ans[i]][j]; sort(ans+1,ans+top+1); printf("%d\n",top); F(i,1,top-1) printf("%d ",ans[i]); printf("%d\n",ans[top]); return 0; }