HDU 5305 Friends (DFS,穷举+剪枝)
题意:
给定n个人,m对朋友关系,如果对于每个人,只能刚好选择其所有朋友中的一半的人进行聊天(只是我和我的朋友,不是我的朋友和我的朋友),那么有多少种情况?只要一个选择不同,视为不同情况。
思路:
比如我在14个朋友中选择了7个跟我聊天,那么另外7人已经完全与我没干系,而和我聊天的7个朋友,也已经和我聊天了,即我们配对了,共7对,他所选择的那一半的人中也必须有我。
其实只考虑所给的m条边就行了。如果是奇数对关系,必定有人是奇数个朋友,那么也就0种情况。如果是偶数条边,还得判断每个人是否都是偶数个朋友,若不是也是0种。
满足了情况之后再对m个关系选取其中的m/2条即可。但是所选的关系也必须是满足要求的,那么对于所选的m/2条关系进行判断即可知道是否满足要求,穷举所有可能进行判断。DFS就可以了,每条边要么选,要么不选。但是必须剪枝才能过。
1 #include <bits/stdc++.h> 2 #define LL long long 3 #define pii pair<int,int> 4 #define INF 0x7f7f7f7f 5 using namespace std; 6 const int N=10; 7 int n, m, s[65], e[65], num, times[N], du[N]; 8 int DFS(int x) //x是第几条边 9 { 10 if( num*2>=m ) //已经够一半了,判断是否满足要求 11 { 12 for(int i=1; i<=n; i++) if( 2*du[i]!= times[i] ) return 0; //每个人的度有一半即可。 13 return 1; 14 } 15 16 int ans=0; 17 if( du[ s[x] ]*2<times[ s[x] ] && du[ e[x] ]*2 <times[e[x]] ) //剪枝:这条边两个端点都已经满度,就不能再选了。 18 { 19 du[s[x]]++,du[e[x]]++; 20 num++; //所选边的数量 21 ans+=DFS(x+1); 22 du[s[x]]--,du[e[x]]--; 23 num--; 24 } 25 26 if( m/2-num < m-x ) //还没有决定是否选的边数必须不小于m的一半 27 ans+=DFS(x+1); 28 return ans; 29 } 30 31 int cal(int n ) 32 { 33 //先检查是否满足奇数度的要求 34 if(m&1) return 0; 35 for(int i=1; i<=n; i++) if( times[i]&1 ) return 0; 36 37 num=0; 38 memset(du,0,sizeof(du)); 39 return DFS(0); 40 } 41 42 int main() 43 { 44 //freopen("e://input.txt", "r", stdin); 45 int t; 46 cin>>t; 47 while(t--) 48 { 49 memset(times, 0, sizeof(times)); 50 scanf( "%d%d",&n,&m ); 51 for(int i=0; i<m; i++) 52 { 53 scanf("%d%d",&s[i],&e[i]); 54 times[s[i] ]++; //记录朋友个数 55 times[e[i] ]++; 56 } 57 printf("%d\n",cal(n)); 58 } 59 return 0; 60 }