码迷,mamicode.com
首页 > 其他好文 > 详细

POJ3565-Ants-KM变形

时间:2020-04-14 14:06:05      阅读:62      评论:0      收藏:0      [点我收藏+]

标签:mic   efi   mem   方案   eof   play   close   open   event   

题意:

给出n个蚂蚁和苹果的坐标,对每个蚂蚁和苹果进行匹配(之间连一条线段),要求:任意两条线段不相交,输出一种方案。

 

思路:

因为三角形的两边之和大于第三边,

所有他们连线的所有线段和最小的时候线段一定不会相交,

所以这道题目就变成了二分图最小权值匹配,把任意两点间权值即为他们的距离,跑一遍KM即可

 

注意:

套KM模板的时候,必须从1开始记录,否则不出结果

出来任意一种方案即可,不一定要和样例输出一样

double类型不要弄错

double类型比较的时候最好用<eps,因为精度问题,直接判断等于0可能会出错

 

AC代码:

技术图片
  1 #include<stdio.h>
  2 #include<iostream>
  3 #include<queue>
  4 #include<cmath>
  5 #include<string.h>
  6 using namespace std;
  7 #define inf 0x3f3f3f3f
  8 typedef long long ll;
  9 const int N=110;
 10 #define inf 0x3f3f3f3f
 11 #define eps 1e-9
 12 
 13 //man x、woman y
 14 int n,match[N];
 15 double love[N][N],exy[N],exx[N],need[N];//***
 16 bool booky[N],bookx[N];
 17 struct node
 18 {
 19     int x,y;
 20 } ant[N],apple[N];
 21 
 22 bool dfs(int woman)
 23 {
 24     booky[woman]=1;
 25     for(int i=1; i<=n; i++)//man
 26     {
 27         if(bookx[i])
 28             continue;
 29         double w=exy[woman]+exx[i]-love[woman][i];//***
 30         if(fabs(w)<=eps )//***
 31             // if(w==0)
 32         {
 33             bookx[i]=1;
 34             if(match[i]==-1||dfs(match[i]))
 35             {
 36                 match[i]=woman;
 37                 return 1;
 38             }
 39         }
 40         else
 41             need[i]=min(need[i],w);
 42     }
 43     return 0;
 44 }
 45 
 46 int KM()
 47 {
 48     for(int i=1; i<=n; i++)
 49     {
 50         exy[i]=inf*(-1.0);//***
 51         for(int j=1; j<=n; j++)
 52             exy[i]=max(exy[i],love[i][j]);
 53     }
 54     for(int i=1; i<=n; i++)
 55     {
 56         fill(need+1,need+n+1,inf);//***
 57         while(1)
 58         {
 59             memset(bookx,0,sizeof bookx);
 60             memset(booky,0,sizeof booky);//放在这里清空,清空和break不能掉头否则TLE
 61             if(dfs(i))
 62                 break;
 63             double minn=inf;//***
 64             for(int j=1; j<=n; j++)
 65             {
 66                 if(bookx[j]==0)
 67                     minn=min(minn,need[j]);
 68             }
 69             for(int j=1; j<=n; j++)
 70             {
 71                 if(booky[j])
 72                     exy[j]-=minn;
 73                 if(bookx[j])
 74                     exx[j]+=minn;
 75                 else
 76                     need[j]-=minn;
 77             }
 78         }
 79     }
 80 //    int sum=0;
 81 //    for(int i=0; i<n; i++)
 82 //        sum+=love[match[i]][i];
 83 //    return sum;
 84 }
 85 
 86 int main()
 87 {
 88     while(~scanf("%d",&n))
 89     {
 90         memset(match,-1,sizeof(match));
 91         memset(exx,0,sizeof exx);
 92         memset(exy,0,sizeof exy);
 93         for(int i=1; i<=n; i++)
 94             scanf("%d %d",&ant[i].x,&ant[i].y);
 95         for(int i=1; i<=n; i++)
 96             scanf("%d %d",&apple[i].x,&apple[i].y);
 97         for(int i=1; i<=n; i++)
 98         {
 99             for(int j=1; j<=n; j++)
100 
101                 love[j][i]=(-1.0)*sqrt((ant[i].x-apple[j].x)*(ant[i].x-apple[j].x)+(ant[i].y-apple[j].y)*(ant[i].y-apple[j].y));
102         }
103         KM();
104         for(int i=1; i<=n; i++)
105             printf("%d\n",match[i]);
106     }
107     return 0;
108 }
View Code

 

POJ3565-Ants-KM变形

标签:mic   efi   mem   方案   eof   play   close   open   event   

原文地址:https://www.cnblogs.com/OFSHK/p/12697586.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!