【题意】有两个人A,B在一个二维平面上捡瓶子放到垃圾箱里,每个人从从初始位置出发,一次只能选择一个瓶子去捡,然后去垃圾箱,再去捡瓶子。。。也可以中途不捡了,给出A,B,垃圾箱和n个瓶子的坐标,求两个人捡完瓶子要的最短距离。
【分析】把距离都求出来,再贪心找到最小距离就好了
【Trick】见代码吧。
【AC代码】
#include <bits/stdc++.h> using namespace std; const int maxn = 100003; struct node{ int id; double val; node(){} node(int id,double val):id(id),val(val){} bool operator<(const node &rhs)const{ return val<rhs.val; } }a[maxn],b[maxn]; double x[maxn],y[maxn]; double get_dis(double x1,double y1,double x2,double y2){ return sqrt((x1-x2)*(x1-x2)+(y1-y2)*(y1-y2)); } double ax,ay,bx,by,tx,ty; int n; int main(){ scanf("%lf%lf%lf%lf%lf%lf",&ax,&ay,&bx,&by,&tx,&ty); scanf("%d",&n); for(int i=1; i<=n; i++) cin>>x[i]>>y[i]; for(int i=1; i<=n; i++){ a[i].id=i; a[i].val = get_dis(x[i],y[i],ax,ay)-get_dis(x[i],y[i],tx,ty); } sort(a+1,a+n+1); //Init for(int i=1; i<=n; i++){ b[i].id=i; b[i].val = get_dis(x[i],y[i],bx,by)-get_dis(x[i],y[i],tx,ty); } sort(b+1,b+n+1); double ans = 0; for(int i=1; i<=n; i++) ans+=2.00*get_dis(x[i],y[i],tx,ty); // cout<<ans<<endl; double temp = ans; a[0]=node{0,0}; b[0]=node{0,0}; ans = min(ans+a[1].val,ans+b[1].val); //本题的Trick点 //必须要有一个人动,才能捡完垃圾 //之前循环里面,有可能ans就是最小的,就一直更新不了,这样就代表没人动,是不行的 for(int ii=0; ii<3; ii++){//A动B不动 for(int jj=0; jj<3; jj++){ if(ii==0&&jj==0)continue; if(a[ii].id==b[jj].id) continue; ans = min(ans,temp+a[ii].val+b[jj].val); } } printf("%.8f\n",ans); return 0; }