日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

POJ - 3565 Ants(二分图最小权匹配+KM+思维)

發(fā)布時間:2024/4/11 编程问答 42 豆豆
生活随笔 收集整理的這篇文章主要介紹了 POJ - 3565 Ants(二分图最小权匹配+KM+思维) 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

題目鏈接:點擊查看

題目大意:給出n個螞蟻和n個蘋果樹的坐標,我們需要求出每個螞蟻平時覓食所要去的蘋果樹,必須保證所有路徑不能有交叉

題目分析:因為所有的邊不能有交叉,所以我們選擇距離最短的兩個點匹配即可,然后n個螞蟻匹配n個蘋果樹,又是一個完備匹配問題,那么這個題目就轉(zhuǎn)換成了一個最小權(quán)匹配的問題了,直接套模板就行了嗎?不,其實還有個很嚴謹?shù)淖C明,一開始我是為了省精度,用了距離的平方建邊,一直WA,后來改成double類型的距離建邊,也就是開了個sqrt然后就A了,一直想不明白為什么,看了大佬的證明后就突然有點小理解了:

這道題能用KM在于四邊形不等式的幾何形式dist (A,B)+dist(C,D)>dist(A,C)+dist(B,D),其中ABCD是四邊形,證明利用對角線交點O與AB,CD兩條邊構(gòu)成的三角形不等式證明,即OA+OB>AB,OC+OD>CD,兩式相加得到AC+BD>AB+CD,對于平方,沒有三角形不等式,事實上可以構(gòu)造使AB*AB+CD*CD=AC*AC+BD*BD

大概就是這樣,求邊權(quán)的時候需要用實際距離,然后因為涉及到了double類型,就要開一個eps保持精度,1e-6足夠

代碼:

#include<iostream> #include<cstdlib> #include<string> #include<cstring> #include<cstdio> #include<algorithm> #include<climits> #include<cmath> #include<cctype> #include<stack> #include<queue> #include<list> #include<vector> #include<set> #include<map> #include<sstream> using namespace std;typedef long long LL;const double inf=1e10;const double eps=1e-6;const int N=110;int n;struct Pos {int x,y; }ant[N],tree[N];double dis(Pos& a,Pos& b) {return sqrt((a.x-b.x)*(a.x-b.x)+(a.y-b.y)*(a.y-b.y)); }double la[N],lb[N];//頂標bool visa[N],visb[N];//訪問標記double maze[N][N];//邊權(quán)int match[N];//右部點匹配了哪一個左部點double upd[N];bool dfs(int x) {visa[x]=true;//訪問標記:x在交錯樹中for(int i=1;i<=n;i++){if(!visb[i]){if(la[x]+lb[i]-maze[x][i]<eps)//相等子圖{visb[i]=true;//訪問標記:y在交錯樹中if(!match[i]||dfs(match[i])){match[i]=x;return true;}}elseupd[i]=min(upd[i],la[x]+lb[i]-maze[x][i]);}}return false; } int KM() {memset(match,0,sizeof(match));for(int i=1;i<=n;i++){la[i]=-inf;lb[i]=0;for(int j=1;j<=n;j++)la[i]=max(la[i],maze[i][j]);}for(int i=1;i<=n;i++){while(1)//直到左部點找到匹配{memset(visa,false,sizeof(visa));memset(visb,false,sizeof(visb));for(int i=1;i<=n;i++)upd[i]=inf;if(dfs(i))break;double delta=inf;for(int j=1;j<=n;j++)if(!visb[j])delta=min(delta,upd[j]);for(int j=1;j<=n;j++)//修改頂標{if(visa[j])la[j]-=delta;if(visb[j])lb[j]+=delta;}}}int ans=0;for(int i=1;i<=n;i++)ans+=maze[match[i]][i];return ans; }int main() { // freopen("input.txt","r",stdin); // ios::sync_with_stdio(false);while(scanf("%d",&n)!=EOF){for(int i=1;i<=n;i++)scanf("%d%d",&ant[i].x,&ant[i].y);for(int i=1;i<=n;i++)scanf("%d%d",&tree[i].x,&tree[i].y);for(int i=1;i<=n;i++)for(int j=1;j<=n;j++)maze[i][j]=-dis(tree[i],ant[j]);KM();for(int i=1;i<=n;i++)printf("%d\n",match[i]);} return 0; }

?

總結(jié)

以上是生活随笔為你收集整理的POJ - 3565 Ants(二分图最小权匹配+KM+思维)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。