我们先来学一下稳定婚姻问题
什么是稳定婚姻问题?
有n个女士和n个男士,他们要一一进行配对。每个男士心中对这n个女士都有一个排名,同理,每个女士心里对n个男性也有一个排名。我们要做的是,在他们配对完成以后,不存在以下这种情况:女士a和男士b进行了配对,但是女士c在男士b心里的排名要高于a,而且男士b在女士c心里的排名也高于女士c当前的伴侣,因为如果出现这种情况男士b和女士c很有可能抛下现在的配对对象而“出轨”。同理女士a也不存在一个男士d,条件也是如此。
我们有一个专门的算法来解决这个问题。算法的大致过程就是男士不断地求婚,而女士不断地拒绝的过程。
在每一轮中,每个单身的男人在没有拒绝过他的女士中找到一个在他心里排名最高的女士进行表白,然后被表白的女士如果此时是单身,那么就会先接受这个男士的表白,暂时和他配对。如果这个女士已经有配对的对象,那么此时这个女士就会比较现在这个向她表白的男士和她现在的配对对象谁再她心里的排名更高,然后她会选择排名更高的那一个拒绝掉另一个。
然后我们再来看这道【LA3989】
在一个盛大的校园舞会上有n位男生和n位女生,每个人都对每个异性有一个排序,代表对他们的喜欢程度。你的任务是将男生和女生一一配对(每个人恰好有一个舞伴),使得男生u和女士v不存在以下情况 1.男生u和女生v不是舞伴;2.他们喜欢对方的程度都大于喜欢各自自己当前舞伴的程度。如果出现了2中的情况,他们可能会擅自抛下自己的舞伴,另外组成一对。
你的任务是对于每个女生,在所有可能和她跳舞的男生中,找出她最喜欢的那个。
这个题就是稳定婚姻问题的模板题了,直接套我们上面的模板的方法就可以。
code from LRJ
1 #include <cstdio> 2 #include <algorithm> 3 #include <iostream> 4 #include <cstring> 5 #include <queue> 6 7 using namespace std; 8 9 const int maxn=1000+10; 10 int pref[maxn][maxn],order[maxn][maxn],Next[maxn]; 11 int future_husband[maxn],future_wife[maxn]; 12 queue<int>q; 13 void engage(int man,int woman){ 14 int m=future_husband[woman]; 15 if(m){ 16 future_wife[m]=0; 17 q.push(m); 18 } 19 future_wife[man]=woman; 20 future_husband[woman]=man; 21 } 22 23 int main(){ 24 int T; 25 scanf("%d",&T); 26 while(T--){ 27 int n; 28 scanf("%d",&n); 29 for(int i=1;i<=n;i++){ 30 for(int j=1;j<=n;j++) 31 scanf("%d",&pref[i][j]);//编号为i的男士第j喜欢的人 32 Next[i]=1; //接下来应该向排名为1的女士求婚 33 future_wife[i]=0; //没有未婚妻 34 q.push(i); 35 } 36 for(int i=1;i<=n;i++){ 37 for(int j=1;j<=n;j++){ 38 int x; 39 scanf("%d",&x); 40 order[i][x]=j; //在编号为i的女士心目中,编号为x的男性的排名 41 } 42 future_husband[i]=0; //没有未婚夫 43 } 44 while(!q.empty()){ 45 int man=q.front();q.pop(); 46 int woman=pref[man][Next[man]++];//下一个求婚对象 47 if(!future_husband[woman]) 48 engage(man,woman); 49 else if(order[woman][man]<order[woman][future_husband[woman]]) 50 engage(man,woman); 51 else q.push(man); 52 } 53 while(!q.empty())q.pop(); 54 for(int i=1;i<=n;i++) printf("%d ",future_wife[i]); 55 if(T)printf(" "); 56 } 57 return 0; 58 }