题目描述
设在一排上有N个格子(N≤20),若在格子中放置有不同颜色的灯,每种灯的个数记为N1,N2,……Nk(k表示不同颜色灯的个数)。
放灯时要遵守下列规则:
①同一种颜色的灯不能分开;
②不同颜色的灯之间至少要有一个空位置。
例如:
N=8(格子数)
R=2(红灯数)
B=3(蓝灯数)
放置的方法有:
R-B顺序
B-R顺序
放置的总数为12种。
程序要求:求排列总数。
输入格式
数据输入的方式为:
N
P1(颜色,为一个字母) N1(灯的数量)
P2 N2
……
Q(结束标记,Q本身不是灯的颜色)
颜色和灯的数量之间由一个空格分隔。
输出
输出排列总数。
样例输入
8
R 2
B 3
Q
样例输出
12
题解:
记录方案的dfs:http://www.cnblogs.com/qscqesze/p/4334157.html
注意同一颜色归到一起
dfs出一种颜色顺序下的所有顺序
再将它乘上N种颜色的排列数N!
即可得到总排列数。。。。
dfs出一种颜色顺序下的所有顺序
再将它乘上N种颜色的排列数N!
即可得到总排列数。。。。
代码:
1 #include <bits/stdc++.h> 2 using namespace std; 3 typedef long long ll; 4 #define mem(a) memset(a,0,sizeof(a)) 5 #define mp(x,y) make_pair(x,y) 6 const int INF = 0x3f3f3f3f; 7 const ll INFLL = 0x3f3f3f3f3f3f3f3fLL; 8 inline ll read(){ 9 ll x=0,f=1;char ch=getchar(); 10 while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();} 11 while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();} 12 return x*f; 13 } 14 ////////////////////////////////////////////////////////////////////////// 15 const int maxn = 1e5+10; 16 17 int n,s,num,k; 18 char c; 19 struct node{ 20 char x; 21 int y; 22 }q[maxn]; 23 24 void dfs(int a,int b,int c){ //a表示已用颜色 b表示已放格子数 c表示是否要放空格 25 if(a==num && b==n){ 26 k++; //同一颜色顺序下排序数 27 return ; 28 } 29 if(b > n) return ; 30 if(!c) dfs(a+1,b+q[a].y,1); 31 dfs(a,b+1,0); 32 } 33 34 int main(){ 35 n = read(); 36 while(cin>>c, c!='Q'){ 37 int b = read(); 38 s += b; 39 bool f = 0; 40 for(int i=0; i<num; i++){ 41 if(q[i].x == c){ 42 q[i].y += b; 43 f = 1; 44 } 45 } 46 if(!f){ 47 q[num].x = c; 48 q[num++].y += b; 49 } 50 } 51 int per=1; 52 for(int i=1; i<=num; i++) 53 per = per*i; 54 if(n - (s+num-1) <= 0) { //格子不够 55 puts("0"); 56 return 0; 57 }else{ 58 dfs(0,0,0); 59 cout << per*k << endl; 60 } 61 62 return 0; 63 }