匈牙利算法
例题1:HDU2063
题目:
RPG girls今天和大家一起去游乐场玩,终于可以坐上梦寐以求的过山车了。可是,过山车的每一排只有两个座位,而且还有条不成文的规矩,就是每个女生必须找个个男生做partner和她同坐。但是,每个女孩都有各自的想法,举个例子把,Rabbit只愿意和XHD或PQK做partner,Grass只愿意和linle或LL做partner,PrincessSnow愿意和水域浪子或伪酷儿做partner。考虑到经费问题,boss刘决定只让找到partner的人去坐过山车,其他的人,嘿嘿,就站在下面看着吧。聪明的Acmer,你可以帮忙算算最多有多少对组合可以坐上过山车吗?
输入:
输入数据的第一行是三个整数K , M , N,分别表示可能的组合数目,女生的人数,男生的人数。0<K<=1000
1<=N 和M<=500.接下来的K行,每行有两个数,分别表示女生Ai愿意和男生Bj做partner。最后一个0结束输入。
输出:
对于每组数据,输出一个整数,表示可以坐上过山车的最多组合数。
样例输入:
6 3 3
1 1
1 2
1 3
2 1
2 3
3 1
0
样例输出:
3
算法解析:他人匈牙利算法借鉴`` HDU2063:

#include<stdio.h>
#include<string.h>
int line[1001][1001];//line[x][y]:x号女和y号男可以相互接受 
int used[1001];//方便递推的时候,进行记录(3号找不到女生,就跟一号女生,一号男生就跟二号女生此时used[]==1;二号男生找到一号,二号女生都名花有主了) 
int partner[1001];//partner[x]=y;x号女对应着y号男 
int w,m;
int findpartner(int x);
int main(){
	int k;
	while(~scanf("%d",&k)&&k){
		getchar();
		int sum=0;
		scanf("%d %d",&w,&m);
		getchar();
		memset(line,0,sizeof(line));
		memset(partner,0,sizeof(partner));//memset,将数组初始化,只能是0或者世-1:哈哈哈 
		for(int i=0;i<k;i++){
			int a,b;
			scanf("%d %d",&a,&b);
			line[a][b]=1;//a,b之间有意愿; 
		}
		for(int i=1;i<=w;i++){
			memset(used,0,sizeof(used));
			if(findpartner(i))
			sum++;
		}
		printf("%d\n",sum);	
	}
	
	return 0;
}
int findpartner(int x){
	for(int i=1;i<=m;i++){
		if(used[i]==0&&line[x][i]==1){
			used[i]=1;
			if(partner[i]==0||findpartner(partner[i])==1){
				partner[i]=x;
				return 1;
			} 
		}
	}
	return 0;
}

例题2:

N个人分配N项任务,一个人只能分配一项任务,一项任务只能分配给一个人,将一项任务分配给一个人是需要支付报酬,如何分配任务,保证支付的报酬总数最小。

解:方法一:实例分析—穷举法

以3个工作人员和3项任务为实例,下图为薪酬图表和根据薪酬图表所得的cost矩阵。

pytorch匈牙利算法 匈牙利算法题目_匈牙利算法


  

利用最简单的方法(穷举法)进行求解,计算出所有分配情况的总薪酬开销,然后求最小值。

total_cost1 = 250 + 600 + 250 = 1100; x00 = 1,x11 = 1,x22 = 1;

total_cost2 = 250 + 350 + 400 = 1000; x00 = 1,x12 = 1,x21 = 1;

total_cost3 = 400 + 400 + 250 = 1050; x01 = 1,x10 = 1,x22 = 1;

total_cost4 = 400 + 350 + 200 = 950; x01 = 1,x12 = 1,x20 = 1; //最优分配

total_cost5 = 350 + 400 + 400 = 1150; x02 = 1,x10 = 1,x21 = 1;

total_cost6 = 350 + 600 + 250 = 1150; x02 = 1,x11 = 1,x22 = 1;

对于任务数和人员数较少时,可利用穷举法计算结果。

若将N任务分配给N个人员,其包含的所有分配情况数目为N!,N增大时,穷举法将难以完成任务。

方法二:匈牙利算法:

其基本的理论基础是针对cost矩阵,将cost矩阵的一行或一列数据加上或减去一个数,其最优任务分配求解问题不变。(cost矩阵:效益矩阵)

pytorch匈牙利算法 匈牙利算法题目_sed_02


【可以去学一下运筹学的知识】

第一步,变换效益矩阵 C ,使每行每列中都有数 0 .每行元素减去本行中最小元素(称为按行减), 每列元

素减去该列中最小元素(称为按列减), 这样就可使每行每列中都有 0 .称所得矩阵为缩减矩阵.

第二步,作出能覆盖所有数 0 的最少水平或竖直的直线.若直线的条数小于方阵 C 的n ,则转入下一步,否则执行第四步 .

第三步 ,进行增 0 变换.在没有被直线覆盖的元素中找出最小元素 ,然后将所有未被直线覆盖的元素减去此最小元素, 所有被二条直线覆盖的交叉处的元素加上此最小元素 ,其余元素不变 .得新的缩减矩阵.返回第二步.

第四步 ,确定缩减矩阵中 n 个独立 0 的位置.对行中只有一个 0 的位置打上“ *” , 同时划去该 0 所在列的其余的0 ;再逐列检查 ,对列中只有一个0 的位置打上“ *” ,同时划去该0 所在行中的其余的0 .如果遇到每行(列)有两个或两个以上的 0 ,则可选择其中一个 0 ,进行处理,确保每行、每列有且仅有一个 0 打上“ *” ,由此找到 n 个独立 0 的位置 ,得最优解的矩阵,从而求出最优 Assignment Problem .

应用举例:

pytorch匈牙利算法 匈牙利算法题目_pytorch匈牙利算法_03


Step1.从第1行减去75,第2行减去35,第3行减去90,第4行减去45

pytorch匈牙利算法 匈牙利算法题目_sed_04


Step2.从第1列减去0,第2列减去0,第3列减去0,第4列减去5。

pytorch匈牙利算法 匈牙利算法题目_匈牙利算法_05


现在已经完成了匈牙利算法的第一部

Step3.利用最少的水平线或垂直线覆盖所有的0。

pytorch匈牙利算法 匈牙利算法题目_sed_06


Step4.由于水平线和垂直线的总数是3,少于4,进入Step5。

现在已经完成了匈牙利算法的第二步

Step5.没有被覆盖的最小值是5,没有被覆盖的每行减去最小值5,被覆盖的每列加上最小值5,然后跳转到步骤3.

pytorch匈牙利算法 匈牙利算法题目_pytorch匈牙利算法_07


匈牙利算法已经完成了第三步

Step3.利用最少的水平线或垂直线覆盖所有的0。

pytorch匈牙利算法 匈牙利算法题目_sed_08


Step4.由于水平线和垂直线的总数是3,少于4,进入Step5。

Step5.没有被覆盖的最小值是20,没有被覆盖的每行减去最小值20,被覆盖的每列加上最小值20,然后跳转到步骤3.

pytorch匈牙利算法 匈牙利算法题目_sed_09


Step3.利用最少的水平线或垂直线覆盖所有的0。

pytorch匈牙利算法 匈牙利算法题目_最小值_10


Step4.由于水平线和垂直线的总数是4,算法结束,分配结果如下图所示。

pytorch匈牙利算法 匈牙利算法题目_匈牙利算法_11


其中,黄色框表示分配结果,左边矩阵的最优分配等价于左边矩阵的最优分配。

L=45+95+55+80;

//本人小白,借鉴了一下大佬的思路,再加上自己的理解。这篇博客是为了捋清楚思路,相当于自己的笔记本,如果强要说是抄袭,本人只能呵呵一笑