BZOJ2425:[HAOI2010]计数(数位DP)

Description

你有一组非零数字(不一定唯一),你可以在其中插入任意个0,这样就可以产生无限个数。比如说给定{1,2},那么可以生成数字12,21,102,120,201,210,1002,1020,等等。

现在给定一个数,问在这个数之前有多少个数。(注意这个数不会有前导0).

Input

只有1行,为1个整数n.

Output

只有整数,表示N之前出现的数的个数。

Sample Input

1020

Sample Output

7

HINT

n的长度不超过50,答案不超过263-1.

Solution

感觉我的数位DP还不是很稳啊……
全靠面向WA和面向样例来编程(逃
首先这个题并不是很难,因为很容易想到:
当某一位没有限制的时候,这一位以及后面的位数就可以用全排列来求数的个数了
求全排列时要去重
只不过数据太大所以求全排列的时候要分解质因数……否则只有60分
啊啊啊好多细节烦死了

Code

 1 #include<iostream>
 2 #include<cstring>
 3 #include<cstdio>
 4 #include<algorithm>
 5 #define LL long long
 6 using namespace std;
 7 LL num[1001],a[1001],cnt,pos,num_sum,Keg[1001];
 8 LL used[1001],prime[50]={0,2,3,5,7,9,11,13,17,19,23,29,31,37,41,43,47};
 9 char n[101];
10 
11 void divide(LL x,LL k)//将x分解质因数加入桶内 
12 {
13     for (LL i=1;i<=16;++i)
14         while (x%prime[i]==0 && x)
15         {
16             Keg[i]+=k;
17             x/=prime[i];
18         }
19 }
20 
21 LL check(LL x)
22 {
23     LL sum=1;
24     memset(Keg,0,sizeof(Keg));//桶清空 
25     for (int i=1;i<=x;++i) divide(i,1);//求全排列 
26     for (LL i=1;i<=9;++i)//若一个数字出现多次就肯定会重复,重复个数为num[i]! 
27     {
28         for (int j=1;j<=num[i];++j) divide(j,-1);
29         x-=num[i];
30     }
31     if (x<0) return 0;//这里的意思是,全排列的位数摆不开那些没有使用的非0数字 
32     for (int i=1;i<=x;++i) divide(i,-1);//别忘了0也要去重 
33     for (int i=1;i<=16;++i)
34         for (int j=1;j<=Keg[i];++j)
35             sum*=prime[i];
36     return sum;
37 }
38 
39 LL Dfs(LL pos,LL limit,LL used)
40 {
41     if (pos==0) return used==num_sum;//必须要所有非0数字用光才能返回1 
42     if (!limit)
43         return check(pos);//没有限制,直接求后面的全排列 
44         
45     LL ans=0,up=limit?a[pos]:9;
46     for (LL i=0;i<=up;++i)
47     {
48         if (num[i]<=0 && i!=0) continue;
49         num[i]--;
50         ans+=Dfs(pos-1,limit && i==a[pos],used+(i!=0));
51         num[i]++;
52     }
53     return ans;
54 }
55 
56 int main()
57 {
58     scanf("%s",n);
59     for (int i=strlen(n)-1;i>=0;--i)
60     {
61         if (n[i]!='0')
62             num[n[i]-'0']++,num_sum++;
63         a[++pos]=n[i]-'0';
64     }
65     printf("%lld",Dfs(pos,true,0)-1);
66 }
posted @ 2018-03-31 08:29  Refun  阅读(240)  评论(0编辑  收藏  举报