二分圖多重最大匹配

( 一 ) 如果x部節(jié)點(diǎn)只對(duì)應(yīng)一個(gè)y部節(jié)點(diǎn),而y部節(jié)點(diǎn)可以對(duì)應(yīng)多個(gè)x部節(jié)點(diǎn),那么這種匹配可以用匈牙利算法來解決。
如何解決?
方法一:
我們知道,傳統(tǒng)的二分匹配是一對(duì)一匹配的,那么我們把y節(jié)點(diǎn)拆點(diǎn),然后再按照正常的二分匹配就可以了。
這樣做的問題是:當(dāng)y節(jié)點(diǎn)很大時(shí),那么拆點(diǎn)耗費(fèi)的時(shí)間會(huì)很多!

方法二:
把y部節(jié)點(diǎn)的match數(shù)組改為二維的,第一維度表示第i個(gè)y節(jié)點(diǎn),第二個(gè)維度表示這個(gè)y節(jié)點(diǎn)剩余的容量

G - Escape
題意:
有N(N<100,000)個(gè)人要去M(M<10)個(gè)星球,每個(gè)人只可以去一個(gè)星球,一個(gè)星球最多容納Ki個(gè)人。請(qǐng)問是否所有人都可以選擇自己的星球
題解:

#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
const int  MAXN=100010;
int cnt[15],cap[15];//cap為y節(jié)點(diǎn)的容量,cnt為當(dāng)前y節(jié)點(diǎn)使用的容量
int graph[MAXN][15];
int n,m,used[15];
int match[15][MAXN];
//match[i][j]=k 第i個(gè)星球上住的第j個(gè)人是k
bool DFS(int x)
{
    for(int i=1;i<=m;i++)
    {
        if(graph[x][i]&&used[i]==0)
        {
            used[i]=1;
            if(cnt[i]<cap[i])//如果當(dāng)前y節(jié)點(diǎn)還有容量可以匹配
            {
                cnt[i]++;
                match[i][cnt[i]]=x;
                return true;
            }
            for(int j=1;j<=cnt[i];j++)//如果y節(jié)點(diǎn)的容量已經(jīng)滿了,嘗試為y節(jié)點(diǎn)的某個(gè)對(duì)象換對(duì)象
            {
                if(DFS(match[i][j]))
                {
                    match[i][j]=x;//y節(jié)點(diǎn)第j個(gè)位置讓給x
                    return true;
                }
            }
        }
    }
    return false;
}
bool judge()
{
    memset(cnt,0,sizeof(cnt));
    for(int i=1;i<=n;i++)//為所有x節(jié)點(diǎn)匹配對(duì)象
    {
        memset(used,0,sizeof(used));
        if(!DFS(i)) return false;
    }
    return true;
}
int main()
{
    while(scanf("%d%d",&n,&m)!=EOF)
    {
        for(int i=1;i<=n;i++)
        {
            for(int j=1;j<=m;j++)
            {
                scanf("%d",&graph[i][j]);
            }
        }
        for(int i=1;i<=m;i++)
        {
            scanf("%d",&cap[i]);
        }
        if(judge()) printf("YES\n");
        else printf("NO\n");
    }
    return 0;
}

( 二 )如果x部節(jié)點(diǎn)可以匹配多個(gè)y部節(jié)點(diǎn),y部節(jié)點(diǎn)可以同時(shí)匹配多個(gè)x部節(jié)點(diǎn),那么應(yīng)該用網(wǎng)絡(luò)流來解決。(因?yàn)樾傺览惴o法應(yīng)對(duì)兩邊都可以選多個(gè)這種情況)
怎么建圖?
很簡(jiǎn)單,假設(shè)x部節(jié)點(diǎn)的容量為capx[ i ],y部節(jié)點(diǎn)的容量為capy[ i ],同時(shí)給出x部節(jié)點(diǎn)可以與y部節(jié)點(diǎn)相連的的邊,那么對(duì)于每個(gè)x部節(jié)點(diǎn),超級(jí)源點(diǎn)都與x部節(jié)點(diǎn)連邊,邊權(quán)為capx[i];對(duì)于每個(gè)y部節(jié)點(diǎn),都與超級(jí)匯點(diǎn)連接邊,邊權(quán)為capy[i]。然后連接每個(gè)x與y直接相連的邊,邊權(quán)為1。

From hihocoder

這樣一來,求出最大流就是最大匹配方案了:流量通道上的邊的剩余流量代表匹配結(jié)果

hidoCoder 1393

#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring>
#include<queue>
using namespace std;
const int MAXN=250;
const int MAXE=20010;
const int INF=0x3f3f3f3f;
struct Node
{
    int to,next,val;
    Node(int to,int next,int val):to(to),next(next),val(val){};
    Node(){}
};
Node edge[MAXE];
int head[MAXN],cnt;
void addEdge(int u,int v,int val)
{
    edge[cnt]=Node(v,head[u],val);head[u]=cnt++;
    edge[cnt]=Node(u,head[v],0);head[v]=cnt++;
}
int step[MAXN];
bool BFS(int st,int ed)
{
    memset(step,-1,sizeof(step));
    step[st]=0;
    queue<int> que;
    que.push(st);
    while(!que.empty())
    {
        int u=que.front();que.pop();
        for(int i=head[u];i!=-1;i=edge[i].next)
        {
            int v=edge[i].to;
            if(edge[i].val>0&&step[v]==-1)
            {
                step[v]=step[u]+1;
                que.push(v);
                if(v==ed) return true;
            }
        }
    }
    return false;
}
int DFS(int st,int ed,int flow)
{
    if(st==ed||flow==0) return flow;
    int curr=0;
    for(int i=head[st];i!=-1;i=edge[i].next)
    {
        int v=edge[i].to;
        if(step[v]==step[st]+1&&edge[i].val>0)
        {
            int d=DFS(v,ed,min(flow,edge[i].val));
            if(d>0)
            {
                edge[i].val-=d;
                edge[i^1].val+=d;
                flow-=d;
                curr+=d;
                if(flow==0) break;
            }
        }
    }
    if(curr==0) step[st]=INF;
    return curr;
}
int Dinic(int st,int ed)
{
    int flow=0;
    while(BFS(st,ed))
    {
        flow+=DFS(st,ed,INF);
    }
    return flow;
}
int main()
{
    int t,n,m,st,ed,need,num,good,item,sum;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d%d",&n,&m);
        sum=cnt=0;st=n+m+1;ed=st+1;
        memset(head,-1,sizeof(head));
        for(int i=1;i<=m;i++)
        {
            scanf("%d",&need);
            sum+=need;
            addEdge(n+i,ed,need);
        }
        for(int i=1;i<=n;i++)
        {
            scanf("%d%d",&num,&good);
            addEdge(st,i,num);
            for(int j=1;j<=good;j++)
            {
                scanf("%d",&item);
                addEdge(i,n+item,1);
            }
        }
        int res=Dinic(st,ed);
        if(res!=sum) printf("No\n");
        else printf("Yes\n");
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,238評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,430評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,134評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,893評(píng)論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,653評(píng)論 6 408
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,136評(píng)論 1 323
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,212評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,372評(píng)論 0 288
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,888評(píng)論 1 334
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,738評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,939評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,482評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,179評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,588評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,829評(píng)論 1 283
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,610評(píng)論 3 391
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,916評(píng)論 2 372

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,781評(píng)論 18 139
  • 中餐: 1.日式鹽烤青花魚 推薦原因:簡(jiǎn)單,好吃(已試,\(^o^)/~) 原料:青花魚2條 檸檬1個(gè) 調(diào)料:海鹽...
    派派小就閱讀 326評(píng)論 0 0
  • 家是卸下偽裝不用小心翼翼的地方,是避風(fēng)的港灣,是倦鳥歸來的巢。而我的朋友,你,就是我的家。我有多大,我們...
    橋妹閱讀 347評(píng)論 0 0
  • 一輪明月 掛在靜謐美好的夜空中 它把夜空為床 ...
    羽love閱讀 614評(píng)論 0 0