Kim Schrijvers
Consider an ordered set S of strings of N (1 <= N <= 31) bits. Bits, of course, are either 0 or 1.
This set of strings is interesting because it is ordered and contains all possible strings of length N that have L (1 <= L <= N) or fewer bits that are `1'.
Your task is to read a number I (1 <= I <= sizeof(S)) from the input and print the Ith element of the ordered set for N bits with no more than L bits that are `1'.
PROGRAM NAME: kimbits
INPUT FORMAT
A single line with three space separated integers: N, L, and I.SAMPLE INPUT (file kimbits.in)
5 3 19
OUTPUT FORMAT
A single line containing the integer that represents the Ith element from the order set, as described.SAMPLE OUTPUT (file kimbits.out)
10011
用组合数缩小范围
比如对于样例中的10000,小于它的有C43+C42+C41+C04个,(右边4个0,选几个变成1)
从左往右一个一个确定是不是1(根据范围,若比它小的数的个数小于题目所求,则这一位为1)
注意取一个1后,L--
#include<cstdio>
#include<cstring>
#include<cmath>
#include<iostream>
#include<algorithm>
#define name "kimbits"
using namespace std;
int C[32][32];
long long n,l,x;
int get(int a,int b)
{
int ret=0;
for (int i=0;i<=b;i++)
ret+=C[a][i];
return ret;
}
int main()
{
freopen(name ".in","r",stdin);
freopen(name ".out","w",stdout);
cin>>n>>l>>x;
int i,j,tmp;
C[1][0]=C[0][0]=C[1][1]=1;
for (i=2;i<=n;i++)
for (j=0;j<=i;j++)
C[i][j]=C[i-1][j]+C[i-1][j-1];
if (x==1) {for (i=1;i<=n;i++)cout<<"0"; cout<<endl;return 0;}
for (i=1;i<=n;i++)
{
tmp=get(i-1,l);
if (tmp>=x) {i--;break;}
}
if (i==n+1) i=n;
tmp=get(i-1,l);
l--;
for (j=i+1;j<=n;j++) cout<<"0";
cout<<"1";
for (j=i-1;j>=1;j--)
{
int g=get(j-1,l);
if (tmp+g>=x) cout<<"0";
else {cout<<"1";l--;tmp+=g;}
}
cout<<endl;
return 0;
}