一、题目
二、解法
首先要确定答案的位数,可以算固定位数不含前导
0
0
0的答案,有一个关键的
d
p
dp
dp,设
d
p
[
i
]
[
j
]
dp[i][j]
dp[i][j]为
i
i
i个字符选
j
j
j个,转移如下:
d
p
[
i
]
[
j
]
=
d
p
[
i
−
1
]
[
j
−
k
]
×
C
(
j
,
k
)
dp[i][j]=dp[i-1][j-k]\times C(j,k)
dp[i][j]=dp[i−1][j−k]×C(j,k)一开始需要先选首位,确定了位数之后,我们再从高到低每一位来确定值,也是用类似的方法,如果方案数够了就可以确定了。
#include <cstdio>
#include <cstring>
#define int long long
const int M = 205;
int read()
{
int num=0,flag=1;
char c;
while((c=getchar())<'0'||c>'9')if(c=='-')flag=-1;
while(c>='0'&&c<='9')num=(num<<3)+(num<<1)+(c^48),c=getchar();
return num*flag;
}
int k,t,len,r[20],ch[20],dp[20][M],C[M][M];
void init()
{
C[0][0]=1;
for(int i=1;i<=200;i++)
{
C[i][0]=1;
for(int j=1;j<=i;j++)
C[i][j]=C[i-1][j]+C[i-1][j-1];
}
}
int dp1(int len)
{
memset(dp,0,sizeof dp);
dp[0][0]=1;
for(int i=1;i<=16;i++)
for(int j=0;j<=len;j++)
for(int k=0;k<=j && k<=r[i-1];k++)
dp[i][j]+=dp[i-1][j-k]*C[j][k];
return dp[16][len];
}
int dp2(int len)
{
int ans=0;
for(int i=0;i<=15;i++)
r[i]=t;
for(int i=1;i<=15;i++)
if(r[i])
{
r[i]--;
ans+=dp1(len-1);
r[i]++;
}
return ans;
}
signed main()
{
init();
k=read();t=read();
for(len=1;;len++)
{
int tmp=dp2(len);
if(tmp>=k) break;
k-=tmp;
}
for(int i=0;i<=15;i++)
r[i]=t;
for(int i=len;i>=1;i--)
{
for(int j=(i==len);j<=15;j++)
{
r[j]--;
int tmp=dp1(i-1);
if(tmp>=k)
{
ch[i]=j;
break;
}
k-=tmp;
r[j]++;
}
}
for(int i=len;i>=1;i--)
if(ch[i]<10) printf("%lld",ch[i]);
else printf("%c",ch[i]+'a'-10);
}